Mercurial > hg > orthanc-webviewer
view Plugin/SeriesVolumeSorter.cpp @ 6:f0042ad844c3 OrthancWebViewer-1.0
notes
author | Sebastien Jodogne <s.jodogne@gmail.com> |
---|---|
date | Thu, 26 Feb 2015 17:28:32 +0100 |
parents | 02f7a0400a91 |
children | a6492d20b2a8 |
line wrap: on
line source
/** * Orthanc - A Lightweight, RESTful DICOM Store * Copyright (C) 2012-2015 Sebastien Jodogne, Medical Physics * Department, University Hospital of Liege, Belgium * * This program is free software: you can redistribute it and/or * modify it under the terms of the GNU Affero General Public License * as published by the Free Software Foundation, either version 3 of * the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see <http://www.gnu.org/licenses/>. **/ #include "SeriesVolumeSorter.h" #include <algorithm> #include <limits> #include <math.h> #include <cassert> namespace OrthancPlugins { SeriesVolumeSorter::SeriesVolumeSorter() : isVolume_(true), sorted_(true) { } void SeriesVolumeSorter::Reserve(size_t countInstances) { positions_.reserve(countInstances); indexes_.reserve(countInstances); } void SeriesVolumeSorter::AddInstance(const std::string& instanceId, const InstanceInformation& instance) { if (instance.HasIndexInSeries()) { indexes_.push_back(std::make_pair(instanceId, instance.GetIndexInSeries())); } if (!isVolume_ || !instance.HasPosition()) { isVolume_ = false; } else { if (positions_.size() == 0) { // This is the first slice in a possible 3D volume. Remember its normal. normal_[0] = instance.GetNormal(0); normal_[1] = instance.GetNormal(1); normal_[2] = instance.GetNormal(2); } else { static const float THRESHOLD = 10.0f * std::numeric_limits<float>::epsilon(); // This is still a possible 3D volume. Check whether the normal // is constant wrt. the previous slices. if (fabs(normal_[0] - instance.GetNormal(0)) > THRESHOLD || fabs(normal_[1] - instance.GetNormal(1)) > THRESHOLD || fabs(normal_[2] - instance.GetNormal(2)) > THRESHOLD) { // The normal is not constant, not a 3D volume. isVolume_ = false; positions_.clear(); } } if (isVolume_) { float distance = (normal_[0] * instance.GetPosition(0) + normal_[1] * instance.GetPosition(1) + normal_[2] * instance.GetPosition(2)); positions_.push_back(std::make_pair(instanceId, distance)); } } sorted_ = false; } std::string SeriesVolumeSorter::GetInstance(size_t index) { if (!sorted_) { if (isVolume_) { assert(indexes_.size() == positions_.size()); std::sort(positions_.begin(), positions_.end(), ComparePosition); float a = positions_.front().second; float b = positions_.back().second; assert(a <= b); if (fabs(b - a) <= 10.0f * std::numeric_limits<float>::epsilon()) { // Not enough difference between the minimum and maximum // positions along the normal of the volume isVolume_ = false; } } if (!isVolume_) { std::sort(indexes_.begin(), indexes_.end(), CompareIndex); } } if (isVolume_) { return positions_[index].first; } else { return indexes_[index].first; } } }