diff --git a/Modules/ROI/autoload/IO/src/mitkROIIO.cpp b/Modules/ROI/autoload/IO/src/mitkROIIO.cpp index 31a8e2c1ee..9cbc5fc05e 100644 --- a/Modules/ROI/autoload/IO/src/mitkROIIO.cpp +++ b/Modules/ROI/autoload/IO/src/mitkROIIO.cpp @@ -1,156 +1,210 @@ /*============================================================================ The Medical Imaging Interaction Toolkit (MITK) Copyright (c) German Cancer Research Center (DKFZ) All rights reserved. Use of this source code is governed by a 3-clause BSD license that can be found in the LICENSE file. ============================================================================*/ #include "mitkROIIO.h" #include <mitkProportionalTimeGeometry.h> #include <mitkROI.h> #include <mitkROIIOMimeTypes.h> -#include <nlohmann/json.hpp> - #include <filesystem> #include <fstream> namespace { - mitk::TimeGeometry::Pointer ReadGeometry(const nlohmann::json& jsonGeometry) + int CheckFileFormat(const nlohmann::json& json) + { + if ("MITK ROI" != json["FileFormat"].get<std::string>()) + mitkThrow() << "Unknown file format (expected \"MITK ROI\")!"; + + auto version = json["Version"].get<int>(); + + if (1 != version) + mitkThrow() << "Unknown file format version (expected version 1)!"; + + return version; + } + + mitk::Vector3D GetSize(const mitk::BaseGeometry* geometry) + { + auto bounds = geometry->GetBounds(); + + mitk::Vector3D result; + result[0] = bounds[1]; + result[1] = bounds[3]; + result[2] = bounds[5]; + + return result; + } + + void SetSize(mitk::BaseGeometry* geometry, const mitk::Vector3D& size) + { + mitk::BaseGeometry::BoundsArrayType bounds({ 0.0, size[0], 0.0, size[1], 0.0, size[2] }); + geometry->SetBounds(bounds); + } + + mitk::TimeGeometry::Pointer ReadGeometry(const nlohmann::json& jGeometry) { auto geometry = mitk::Geometry3D::New(); geometry->ImageGeometryOn(); - if (!jsonGeometry.is_object()) + if (!jGeometry.is_object()) mitkThrow() << "Geometry is expected to be a JSON object."; - if (jsonGeometry.contains("Origin")) - geometry->SetOrigin(jsonGeometry["Origin"].get<mitk::Point3D>()); + if (jGeometry.contains("Origin")) + geometry->SetOrigin(jGeometry["Origin"].get<mitk::Point3D>()); - if (jsonGeometry.contains("Spacing")) - geometry->SetSpacing(jsonGeometry["Spacing"].get<mitk::Vector3D>()); + if (jGeometry.contains("Spacing")) + geometry->SetSpacing(jGeometry["Spacing"].get<mitk::Vector3D>()); - if (jsonGeometry.contains("Size")) - { - auto size = jsonGeometry["Size"].get<mitk::Vector3D>(); - mitk::BaseGeometry::BoundsArrayType bounds({ 0.0, size[0], 0.0, size[1], 0.0, size[2] }); - geometry->SetBounds(bounds); - } + if (jGeometry.contains("Size")) + SetSize(geometry, jGeometry["Size"].get<mitk::Vector3D>()); - auto timeSteps = jsonGeometry.contains("TimeSteps") - ? jsonGeometry["TimeSteps"].get<mitk::TimeStepType>() + auto timeSteps = jGeometry.contains("TimeSteps") + ? jGeometry["TimeSteps"].get<mitk::TimeStepType>() : 1; auto result = mitk::ProportionalTimeGeometry::New(); result->Initialize(geometry, timeSteps); return result; } + + nlohmann::json WriteGeometry(const mitk::TimeGeometry* timeGeometry) + { + auto geometry = timeGeometry->GetGeometryForTimeStep(0); + + nlohmann::json result = { + { "Origin", geometry->GetOrigin() }, + { "Spacing", geometry->GetSpacing() }, + { "Size", GetSize(geometry) } + }; + + auto timeSteps = timeGeometry->CountTimeSteps(); + + if (timeSteps > 1) + result["TimeSteps"] = timeSteps; + + return result; + } } mitk::ROIIO::ROIIO() : AbstractFileIO(ROI::GetStaticNameOfClass(), MitkROIIOMimeTypes::ROI_MIMETYPE(), "MITK ROI") { this->RegisterService(); } std::vector<mitk::BaseData::Pointer> mitk::ROIIO::DoRead() { auto *stream = this->GetInputStream(); std::ifstream fileStream; if (nullptr == stream) { auto filename = this->GetInputLocation(); if (filename.empty() || !std::filesystem::exists(filename)) mitkThrow() << "Invalid or nonexistent filename: \"" << filename << "\"!"; fileStream.open(filename); if (!fileStream.is_open()) mitkThrow() << "Could not open file \"" << filename << "\" for reading!"; stream = &fileStream; } auto result = ROI::New(); try { - auto json = nlohmann::json::parse(*stream); + auto j = nlohmann::json::parse(*stream); - if ("MITK ROI" != json["FileFormat"].get<std::string>()) - mitkThrow() << "Unknown file format (expected \"MITK ROI\")!"; + CheckFileFormat(j); - if (1 != json["Version"].get<int>()) - mitkThrow() << "Unknown file format version (expected version 1)!"; + auto geometry = ReadGeometry(j["Geometry"]); + result->SetTimeGeometry(geometry); + + if (j.contains("Name")) + result->SetProperty("name", mitk::StringProperty::New(j["Name"].get<std::string>())); - result->SetTimeGeometry(ReadGeometry(json["Geometry"])); - - if (json.contains("Properties")) - { - auto properties = mitk::PropertyList::New(); - properties->FromJSON(json["Properties"]); - result->GetPropertyList()->ConcatenatePropertyList(properties); - } - - for (const auto& jsonROI : json["ROIs"]) - { - ROI::Element roi(jsonROI["ID"].get<unsigned int>()); - - if (jsonROI.contains("TimeSteps")) - { - for (const auto& jsonTimeStep : jsonROI["TimeSteps"]) - { - auto t = jsonTimeStep["t"].get<TimeStepType>(); - - roi.SetMin(jsonTimeStep["Min"].get<Point3D>(), t); - roi.SetMax(jsonTimeStep["Max"].get<Point3D>(), t); - - if (jsonTimeStep.contains("Properties")) - { - auto properties = mitk::PropertyList::New(); - properties->FromJSON(jsonTimeStep["Properties"]); - roi.SetProperties(properties, t); - } - } - } - else - { - roi.SetMin(jsonROI["Min"].get<Point3D>()); - roi.SetMax(jsonROI["Max"].get<Point3D>()); - } - - if (jsonROI.contains("Properties")) - { - auto properties = mitk::PropertyList::New(); - properties->FromJSON(jsonROI["Properties"]); - roi.SetDefaultProperties(properties); - } - - result->AddElement(roi); - } + if (j.contains("Caption")) + result->SetProperty("caption", mitk::StringProperty::New(j["Caption"].get<std::string>())); + + for (const auto& roi : j["ROIs"]) + result->AddElement(roi.get<ROI::Element>()); } catch (const nlohmann::json::exception &e) { mitkThrow() << e.what(); } return { result }; } void mitk::ROIIO::Write() { + auto input = dynamic_cast<const ROI*>(this->GetInput()); + + if (input == nullptr) + mitkThrow() << "Invalid input for writing!"; + + if (input->GetNumberOfElements() == 0) + mitkThrow() << "No ROIs found!"; + + auto* stream = this->GetOutputStream(); + std::ofstream fileStream; + + if (stream == nullptr) + { + auto filename = this->GetOutputLocation(); + + if (filename.empty()) + mitkThrow() << "Neither an output stream nor an output filename was specified!"; + + fileStream.open(filename); + + if (!fileStream.is_open()) + mitkThrow() << "Could not open file \"" << filename << "\" for writing!"; + + stream = &fileStream; + } + + if (!stream->good()) + mitkThrow() << "Stream for writing is not good!"; + + nlohmann::ordered_json j = { + { "FileFormat", "MITK ROI" }, + { "Version", 1 }, + { "Name", input->GetProperty("name")->GetValueAsString() }, + { "Geometry", WriteGeometry(input->GetTimeGeometry()) } + }; + + auto caption = input->GetConstProperty("caption"); + + if (caption.IsNotNull()) + j["Caption"] = caption->GetValueAsString(); + + nlohmann::json rois; + + for (const auto& roi : *input) + rois.push_back(roi.second); + + j["ROIs"] = rois; + + *stream << std::setw(2) << j << std::endl; } mitk::ROIIO* mitk::ROIIO::IOClone() const { return new ROIIO(*this); } diff --git a/Modules/ROI/include/mitkROI.h b/Modules/ROI/include/mitkROI.h index 0e76d25c22..d580f5f9b5 100644 --- a/Modules/ROI/include/mitkROI.h +++ b/Modules/ROI/include/mitkROI.h @@ -1,104 +1,109 @@ /*============================================================================ The Medical Imaging Interaction Toolkit (MITK) Copyright (c) German Cancer Research Center (DKFZ) All rights reserved. Use of this source code is governed by a 3-clause BSD license that can be found in the LICENSE file. ============================================================================*/ #ifndef mitkROI_h #define mitkROI_h #include <mitkBaseData.h> #include <MitkROIExports.h> namespace mitk { class MITKROI_EXPORT ROI : public BaseData { public: class MITKROI_EXPORT Element : public IPropertyOwner { public: using PointsType = std::map<TimeStepType, Point3D>; using PropertyListsType = std::map<TimeStepType, PropertyList::Pointer>; Element(); explicit Element(unsigned int id); ~Element() = default; BaseProperty::ConstPointer GetConstProperty(const std::string& propertyKey, const std::string& contextName = "", bool fallBackOnDefaultContext = true) const override; std::vector<std::string> GetPropertyKeys(const std::string& contextName = "", bool includeDefaultContext = false) const override; std::vector<std::string> GetPropertyContextNames() const override; BaseProperty* GetNonConstProperty(const std::string& propertyKey, const std::string& contextName = "", bool fallBackOnDefaultContext = true) override; void SetProperty(const std::string& propertyKey, BaseProperty* property, const std::string& contextName = "", bool fallBackOnDefaultContext = false) override; void RemoveProperty(const std::string& propertyKey, const std::string& contextName = "", bool fallBackOnDefaultContext = false) override; unsigned int GetID() const; void SetID(unsigned int id); bool HasTimeStep(TimeStepType t) const; + bool HasTimeSteps() const; + std::vector<TimeStepType> GetTimeSteps() const; Point3D GetMin(TimeStepType t = 0) const; void SetMin(const Point3D& min, TimeStepType t = 0); Point3D GetMax(TimeStepType t = 0) const; void SetMax(const Point3D& max, TimeStepType t = 0); PropertyList* GetDefaultProperties() const; void SetDefaultProperties(PropertyList* properties); PropertyList* GetProperties(TimeStepType t = 0) const; void SetProperties(PropertyList* properties, TimeStepType t = 0); private: unsigned int m_ID; PointsType m_Min; PointsType m_Max; PropertyList::Pointer m_DefaultProperties; PropertyListsType m_Properties; }; mitkClassMacro(ROI, BaseData) itkFactorylessNewMacro(Self) itkCloneMacro(Self) using ElementsType = std::map<unsigned int, Element>; using Iterator = ElementsType::iterator; using ConstIterator = ElementsType::const_iterator; size_t GetNumberOfElements() const; void AddElement(const Element& element); const Element& GetElement(unsigned int id) const; Element& GetElement(unsigned int id); ConstIterator begin() const; ConstIterator end() const; Iterator begin(); Iterator end(); void SetRequestedRegionToLargestPossibleRegion() override; bool RequestedRegionIsOutsideOfTheBufferedRegion() override; bool VerifyRequestedRegion() override; void SetRequestedRegion(const itk::DataObject* data) override; protected: mitkCloneMacro(Self) ROI(); ROI(const Self& other); ~ROI() override; private: ElementsType m_Elements; }; + + MITKROI_EXPORT void to_json(nlohmann::json& j, const ROI::Element& roi); + MITKROI_EXPORT void from_json(const nlohmann::json& j, ROI::Element& roi); } #endif diff --git a/Modules/ROI/src/mitkROI.cpp b/Modules/ROI/src/mitkROI.cpp index 7243726046..263392da58 100644 --- a/Modules/ROI/src/mitkROI.cpp +++ b/Modules/ROI/src/mitkROI.cpp @@ -1,282 +1,369 @@ /*============================================================================ The Medical Imaging Interaction Toolkit (MITK) Copyright (c) German Cancer Research Center (DKFZ) All rights reserved. Use of this source code is governed by a 3-clause BSD license that can be found in the LICENSE file. ============================================================================*/ #include <mitkROI.h> +void mitk::to_json(nlohmann::json& j, const ROI::Element& roi) +{ + j["ID"] = roi.GetID(); + + if (roi.HasTimeSteps()) + { + auto timeSteps = roi.GetTimeSteps(); + + for (const auto t : timeSteps) + { + nlohmann::json jTimeStep = { + { "t", t }, + { "Min", roi.GetMin(t) }, + { "Max", roi.GetMax(t) } + }; + + if (auto* properties = roi.GetProperties(t); properties != nullptr && !properties->IsEmpty()) + properties->ToJSON(jTimeStep["Properties"]); + + j["TimeSteps"].push_back(jTimeStep); + } + } + else + { + j["Min"] = roi.GetMin(); + j["Max"] = roi.GetMax(); + } + + if (auto* properties = roi.GetDefaultProperties(); properties != nullptr && !properties->IsEmpty()) + properties->ToJSON(j["Properties"]); +} + +void mitk::from_json(const nlohmann::json& j, ROI::Element& roi) +{ + auto id = j["ID"].get<unsigned int>(); + roi.SetID(id); + + if (j.contains("TimeSteps")) + { + for (const auto& jTimeStep : j["TimeSteps"]) + { + auto t = jTimeStep["t"].get<TimeStepType>(); + + roi.SetMin(jTimeStep["Min"].get<Point3D>(), t); + roi.SetMax(jTimeStep["Max"].get<Point3D>(), t); + + if (jTimeStep.contains("Properties")) + { + auto properties = mitk::PropertyList::New(); + properties->FromJSON(jTimeStep["Properties"]); + roi.SetProperties(properties, t); + } + } + } + else + { + roi.SetMin(j["Min"].get<Point3D>()); + roi.SetMax(j["Max"].get<Point3D>()); + } + + if (j.contains("Properties")) + { + auto properties = mitk::PropertyList::New(); + properties->FromJSON(j["Properties"]); + roi.SetDefaultProperties(properties); + } +} + mitk::ROI::Element::Element() : Element(0) { } mitk::ROI::Element::Element(unsigned int id) : m_ID(id), m_DefaultProperties(PropertyList::New()) { } unsigned int mitk::ROI::Element::GetID() const { return m_ID; } void mitk::ROI::Element::SetID(unsigned int id) { m_ID = id; } bool mitk::ROI::Element::HasTimeStep(TimeStepType t) const { return m_Min.count(t) != 0 && m_Max.count(t) != 0; } +bool mitk::ROI::Element::HasTimeSteps() const +{ + return m_Min.size() > 1 && m_Max.size() > 1; +} + +std::vector<mitk::TimeStepType> mitk::ROI::Element::GetTimeSteps() const +{ + std::vector<TimeStepType> result; + result.reserve(m_Min.size()); + + for (const auto& [t, min] : m_Min) + { + if (m_Max.count(t) != 0) + result.push_back(t); + } + + return result; +} + mitk::Point3D mitk::ROI::Element::GetMin(TimeStepType t) const { return m_Min.at(t); } void mitk::ROI::Element::SetMin(const Point3D& min, TimeStepType t) { m_Min[t] = min; } mitk::Point3D mitk::ROI::Element::GetMax(TimeStepType t) const { return m_Max.at(t); } void mitk::ROI::Element::SetMax(const Point3D& max, TimeStepType t) { m_Max[t] = max; } mitk::PropertyList* mitk::ROI::Element::GetDefaultProperties() const { return m_DefaultProperties; } void mitk::ROI::Element::SetDefaultProperties(PropertyList* properties) { m_DefaultProperties = properties; } mitk::PropertyList* mitk::ROI::Element::GetProperties(TimeStepType t) const { if (m_Properties.count(t) != 0) return m_Properties.at(t); return nullptr; } void mitk::ROI::Element::SetProperties(PropertyList* properties, TimeStepType t) { m_Properties[t] = properties; } mitk::BaseProperty::ConstPointer mitk::ROI::Element::GetConstProperty(const std::string& propertyKey, const std::string& contextName, bool fallBackOnDefaultContext) const { if (!contextName.empty()) { const TimeStepType t = std::stoul(contextName); auto it = m_Properties.find(t); if (it != m_Properties.end() && it->second.IsNotNull()) { auto property = it->second->GetConstProperty(propertyKey); if (property.IsNotNull()) return property; } if (!fallBackOnDefaultContext) return nullptr; } return m_DefaultProperties->GetConstProperty(propertyKey); } std::vector<std::string> mitk::ROI::Element::GetPropertyKeys(const std::string& contextName, bool includeDefaultContext) const { if (!contextName.empty()) { const TimeStepType t = std::stoul(contextName); auto it = m_Properties.find(t); std::vector<std::string> result; if (it != m_Properties.end() && it->second.IsNotNull()) result = it->second->GetPropertyKeys(); if (includeDefaultContext) { auto keys = m_DefaultProperties->GetPropertyKeys(); auto end = result.cend(); std::remove_copy_if(keys.cbegin(), keys.cend(), std::back_inserter(result), [&, result, end](const std::string& key) { return end != std::find(result.cbegin(), end, key); }); } return result; } return m_DefaultProperties->GetPropertyKeys(); } std::vector<std::string> mitk::ROI::Element::GetPropertyContextNames() const { std::vector<std::string> result; result.reserve(m_Properties.size()); std::transform(m_Properties.cbegin(), m_Properties.cend(), std::back_inserter(result), [](const PropertyListsType::value_type& property) { return std::to_string(property.first); }); return result; } mitk::BaseProperty* mitk::ROI::Element::GetNonConstProperty(const std::string& propertyKey, const std::string& contextName, bool fallBackOnDefaultContext) { if (!contextName.empty()) { const TimeStepType t = std::stoul(contextName); auto it = m_Properties.find(t); if (it != m_Properties.end() && it->second.IsNotNull()) { auto* property = it->second->GetNonConstProperty(propertyKey); if (property != nullptr) return property; } if (!fallBackOnDefaultContext) return nullptr; } return m_DefaultProperties->GetNonConstProperty(propertyKey); } void mitk::ROI::Element::SetProperty(const std::string& propertyKey, BaseProperty* property, const std::string& contextName, bool fallBackOnDefaultContext) { if (!contextName.empty()) { const TimeStepType t = std::stoul(contextName); auto it = m_Properties.find(t); if (it != m_Properties.end() && it->second.IsNotNull()) { it->second->SetProperty(propertyKey, property); } else if (!fallBackOnDefaultContext) { mitkThrow() << "Context \"" << contextName << "\" does not exist!"; } } m_DefaultProperties->SetProperty(propertyKey, property); } void mitk::ROI::Element::RemoveProperty(const std::string& propertyKey, const std::string& contextName, bool fallBackOnDefaultContext) { if (!contextName.empty()) { const TimeStepType t = std::stoul(contextName); auto it = m_Properties.find(t); if (it != m_Properties.end() && it->second.IsNotNull()) { it->second->RemoveProperty(propertyKey); } else if (!fallBackOnDefaultContext) { mitkThrow() << "Context \"" << contextName << "\" does not exist!"; } } m_DefaultProperties->RemoveProperty(propertyKey); } mitk::ROI::ROI() { } mitk::ROI::ROI(const Self& other) : BaseData(other) { } mitk::ROI::~ROI() { } size_t mitk::ROI::GetNumberOfElements() const { return m_Elements.size(); } void mitk::ROI::AddElement(const Element& element) { const auto id = element.GetID(); if (m_Elements.count(id) != 0) mitkThrow() << "ROI already contains an element with ID " << std::to_string(id) << '.'; m_Elements[id] = element; } const mitk::ROI::Element& mitk::ROI::GetElement(unsigned int id) const { return m_Elements.at(id); } mitk::ROI::Element& mitk::ROI::GetElement(unsigned int id) { return m_Elements.at(id); } mitk::ROI::ConstIterator mitk::ROI::begin() const { return m_Elements.begin(); } mitk::ROI::ConstIterator mitk::ROI::end() const { return m_Elements.end(); } mitk::ROI::Iterator mitk::ROI::begin() { return m_Elements.begin(); } mitk::ROI::Iterator mitk::ROI::end() { return m_Elements.end(); } void mitk::ROI::SetRequestedRegionToLargestPossibleRegion() { } bool mitk::ROI::RequestedRegionIsOutsideOfTheBufferedRegion() { return false; } bool mitk::ROI::VerifyRequestedRegion() { return true; } void mitk::ROI::SetRequestedRegion(const itk::DataObject* data) { }