diff --git a/Framework/Geometry/CMakeLists.txt b/Framework/Geometry/CMakeLists.txt index d161a89ad980432a7ccc2be639d08ab73589e763..6ff724752f42acb983bdd53c9e4d6e2dc9a059f9 100644 --- a/Framework/Geometry/CMakeLists.txt +++ b/Framework/Geometry/CMakeLists.txt @@ -49,6 +49,7 @@ set ( SRC_FILES src/Instrument/ComponentHelper.cpp src/Instrument/ComponentInfo.cpp src/Instrument/ComponentInfoBankHelpers.cpp + src/Instrument/ComponentInfoIterator.cpp src/Instrument/Container.cpp src/Instrument/Detector.cpp src/Instrument/DetectorGroup.cpp @@ -106,8 +107,8 @@ set ( SRC_FILES src/Objects/BoundingBox.cpp src/Objects/CSGObject.cpp src/Objects/InstrumentRayTracer.cpp - src/Objects/MeshObject2D.cpp src/Objects/MeshObject.cpp + src/Objects/MeshObject2D.cpp src/Objects/MeshObjectCommon.cpp src/Objects/RuleItems.cpp src/Objects/Rules.cpp @@ -198,6 +199,8 @@ set ( INC_FILES inc/MantidGeometry/Instrument/ComponentHelper.h inc/MantidGeometry/Instrument/ComponentInfo.h inc/MantidGeometry/Instrument/ComponentInfoBankHelpers.h + inc/MantidGeometry/Instrument/ComponentInfoItem.h + inc/MantidGeometry/Instrument/ComponentInfoIterator.h inc/MantidGeometry/Instrument/ComponentVisitor.h inc/MantidGeometry/Instrument/Container.h inc/MantidGeometry/Instrument/Detector.h @@ -207,9 +210,10 @@ set ( INC_FILES inc/MantidGeometry/Instrument/DetectorInfoIterator.h inc/MantidGeometry/Instrument/FitParameter.h inc/MantidGeometry/Instrument/Goniometer.h - inc/MantidGeometry/Instrument/GridDetector.h + inc/MantidGeometry/Instrument/GridDetector.h inc/MantidGeometry/Instrument/GridDetectorPixel.h inc/MantidGeometry/Instrument/IDFObject.h + inc/MantidGeometry/Instrument/InfoIteratorBase.h inc/MantidGeometry/Instrument/InstrumentDefinitionParser.h inc/MantidGeometry/Instrument/InstrumentVisitor.h inc/MantidGeometry/Instrument/ObjCompAssembly.h @@ -307,6 +311,7 @@ set ( TEST_FILES CenteringGroupTest.h CompAssemblyTest.h ComponentInfoBankHelpersTest.h + ComponentInfoIteratorTest.h ComponentInfoTest.h ComponentParserTest.h ComponentTest.h @@ -327,8 +332,8 @@ set ( TEST_FILES GeneralFrameTest.h GeneralTest.h GoniometerTest.h - GridDetectorTest.h GridDetectorPixelTest.h + GridDetectorTest.h GroupTest.h GroupTransformationTest.h HKLFilterTest.h diff --git a/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfo.h b/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfo.h index fcfc5e8c1e21f76be3dc73c7260a286cfe4ca6c4..15fefd005c235c0769389b66b7d49d69437ca8a2 100644 --- a/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfo.h +++ b/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfo.h @@ -9,6 +9,7 @@ #include "MantidBeamline/ComponentType.h" #include "MantidGeometry/DllConfig.h" +#include "MantidGeometry/Instrument/ComponentInfoIterator.h" #include "MantidGeometry/Objects/BoundingBox.h" #include "MantidKernel/DateAndTime.h" #include <boost/shared_ptr.hpp> @@ -140,9 +141,18 @@ public: Types::Core::DateAndTime> &interval); size_t scanCount() const; void merge(const ComponentInfo &other); + + ComponentInfoIterator<ComponentInfo> begin(); + ComponentInfoIterator<ComponentInfo> end(); + const ComponentInfoIterator<const ComponentInfo> cbegin(); + const ComponentInfoIterator<const ComponentInfo> cend(); + friend class Instrument; }; +using ComponentInfoIt = ComponentInfoIterator<ComponentInfo>; +using ComponentInfoConstIt = ComponentInfoIterator<const ComponentInfo>; + } // namespace Geometry } // namespace Mantid diff --git a/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfoItem.h b/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfoItem.h new file mode 100644 index 0000000000000000000000000000000000000000..2814998359dec3a106e87d7b00b3ed97c2c2a611 --- /dev/null +++ b/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfoItem.h @@ -0,0 +1,55 @@ +// Mantid Repository : https://github.com/mantidproject/mantid +// +// Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI, +// NScD Oak Ridge National Laboratory, European Spallation Source +// & Institut Laue - Langevin +// SPDX - License - Identifier: GPL - 3.0 + +#ifndef MANTID_GEOMETRY_COMPONENTINFOITEM_H_ +#define MANTID_GEOMETRY_COMPONENTINFOITEM_H_ + +#include <MantidKernel/Quat.h> +#include <MantidKernel/V3D.h> +#include <vector> + +namespace Mantid { +namespace Geometry { + +/** ComponentInfoItem + * Return type for ComponentInfoIterators. Provides AOS type access to + * ComponentInfo + */ +template <typename T> class ComponentInfoItem { +public: + ComponentInfoItem(T &componentInfo, const size_t index) + : m_componentInfo(&componentInfo), m_index(index) {} + + bool isDetector() const { return m_componentInfo->isDetector(m_index); } + std::vector<size_t> detectorsInSubtree() const { + return m_componentInfo->detectorsInSubtree(m_index); + } + std::vector<size_t> componentsInSubtree() const { + return m_componentInfo->componentsInSubtree(m_index); + } + const std::vector<size_t> &children() const { + return m_componentInfo->children(m_index); + } + Kernel::V3D position() const { return m_componentInfo->position(m_index); } + Kernel::Quat rotation() const { return m_componentInfo->rotation(m_index); } + size_t parent() const { return m_componentInfo->parent(m_index); } + bool hasParent() const { return m_componentInfo->hasParent(m_index); } + Kernel::V3D scaleFactor() const { + return m_componentInfo->scaleFactor(m_index); + } + std::string name() const { return m_componentInfo->name(m_index); } + size_t index() const { return m_index; } + + // Non-owning pointer. A reference makes the class unable to define an + // assignment operator that we need. + T *m_componentInfo; + size_t m_index; +}; + +} // namespace Geometry +} // namespace Mantid + +#endif /* MANTID_GEOMETRY_COMPONENTINFOITEM_H_ */ diff --git a/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfoIterator.h b/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfoIterator.h new file mode 100644 index 0000000000000000000000000000000000000000..949bc55a7c59844d2678a93b4638ccef09b0bea0 --- /dev/null +++ b/Framework/Geometry/inc/MantidGeometry/Instrument/ComponentInfoIterator.h @@ -0,0 +1,26 @@ +// Mantid Repository : https://github.com/mantidproject/mantid +// +// Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI, +// NScD Oak Ridge National Laboratory, European Spallation Source +// & Institut Laue - Langevin +// SPDX - License - Identifier: GPL - 3.0 + +#ifndef MANTID_GEOMETRY_COMPONENTINFOITERATOR_H_ +#define MANTID_GEOMETRY_COMPONENTINFOITERATOR_H_ + +#include "MantidGeometry/Instrument/ComponentInfoItem.h" +#include "MantidGeometry/Instrument/InfoIteratorBase.h" + +namespace Mantid { +namespace Geometry { + +/** ComponentInfoIterator for random access iteration over ComponentInfo + */ +template <typename T> +class ComponentInfoIterator : public InfoIteratorBase<T, ComponentInfoItem> { +public: + using InfoIteratorBase<T, ComponentInfoItem>::InfoIteratorBase; +}; +} // namespace Geometry +} // namespace Mantid + +#endif /* MANTID_GEOMETRY_COMPONENTINFOITERATOR_H_ */ diff --git a/Framework/Geometry/inc/MantidGeometry/Instrument/DetectorInfoItem.h b/Framework/Geometry/inc/MantidGeometry/Instrument/DetectorInfoItem.h index 3699f43492230bf8f19cd3c5a948a755f55ce83e..659fa2804a03d28b3456400e7f45bdcf308f366a 100644 --- a/Framework/Geometry/inc/MantidGeometry/Instrument/DetectorInfoItem.h +++ b/Framework/Geometry/inc/MantidGeometry/Instrument/DetectorInfoItem.h @@ -53,6 +53,10 @@ public: return m_detectorInfo->rotation(m_index); } + double l2() const { return m_detectorInfo->l2(m_index); } + + size_t index() const { return m_index; } + DetectorInfoItem(T &detectorInfo, const size_t index) : m_detectorInfo(&detectorInfo), m_index(index) {} diff --git a/Framework/Geometry/inc/MantidGeometry/Instrument/DetectorInfoIterator.h b/Framework/Geometry/inc/MantidGeometry/Instrument/DetectorInfoIterator.h index 3b40de2d809b94523fa02e271e7fc506ab2ff450..4c604e26da0a9274ca2cf762263056baa4665e0a 100644 --- a/Framework/Geometry/inc/MantidGeometry/Instrument/DetectorInfoIterator.h +++ b/Framework/Geometry/inc/MantidGeometry/Instrument/DetectorInfoIterator.h @@ -8,80 +8,22 @@ #define MANTID_GEOMETRY_DETECTORINFOITERATOR_H_ #include "MantidGeometry/Instrument/DetectorInfoItem.h" -#include <boost/iterator/iterator_facade.hpp> - -using Mantid::Geometry::DetectorInfoItem; +#include "MantidGeometry/Instrument/InfoIteratorBase.h" namespace Mantid { namespace Geometry { -/** DetectorInfoIterator - -DetectorInfoIterator allows users of the DetectorInfo object access to data -via an iterator. The iterator works as a slice view in that the index is -incremented and all items accessible at that index are made available via the -iterator. +/** DetectorInfoIterator for random access iteration over DetectorInfo +Random access iterator for iteration over DetectorInfo @author Bhuvan Bezawada, STFC @date 2018 */ template <typename T> -class DetectorInfoIterator - : public boost::iterator_facade<DetectorInfoIterator<T>, - DetectorInfoItem<T> &, - boost::random_access_traversal_tag> { - +class DetectorInfoIterator : public InfoIteratorBase<T, DetectorInfoItem> { public: - DetectorInfoIterator(T &detectorInfo, const size_t index) - : m_item(detectorInfo, index) {} - -private: - // Allow boost iterator access - friend class boost::iterator_core_access; - - // Iterator methods - void advance(int64_t delta) { - m_item.m_index = - delta < 0 ? std::max(static_cast<uint64_t>(0), - static_cast<uint64_t>(m_item.m_index) + delta) - : std::min(m_item.m_detectorInfo->size(), - m_item.m_index + static_cast<size_t>(delta)); - } - - // This could cause a segmentation fault if a user goes past the end of the - // iterator and tries to index into the n+1 th element (which would not - // exist). Adding range checks to all the above methods may slow down - // performance though. - void increment() { - if (m_item.m_index < m_item.m_detectorInfo->size()) { - ++m_item.m_index; - } - } - - void decrement() { - if (m_item.m_index > 0) { - --m_item.m_index; - } - } - - size_t getIndex() const { return m_item.m_index; } - - void setIndex(const size_t index) { m_item.m_index = index; } - - bool equal(const DetectorInfoIterator<T> &other) const { - return getIndex() == other.getIndex(); - } - - DetectorInfoItem<T> &dereference() const { return m_item; } - - uint64_t distance_to(const DetectorInfoIterator<T> &other) const { - return static_cast<uint64_t>(other.getIndex()) - - static_cast<uint64_t>(getIndex()); - } - - mutable DetectorInfoItem<T> m_item; + using InfoIteratorBase<T, DetectorInfoItem>::InfoIteratorBase; }; - } // namespace Geometry } // namespace Mantid diff --git a/Framework/Geometry/inc/MantidGeometry/Instrument/InfoIteratorBase.h b/Framework/Geometry/inc/MantidGeometry/Instrument/InfoIteratorBase.h new file mode 100644 index 0000000000000000000000000000000000000000..9e82ae163573aab3475489607d36f0635601af1a --- /dev/null +++ b/Framework/Geometry/inc/MantidGeometry/Instrument/InfoIteratorBase.h @@ -0,0 +1,90 @@ +// Mantid Repository : https://github.com/mantidproject/mantid +// +// Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI, +// NScD Oak Ridge National Laboratory, European Spallation Source +// & Institut Laue - Langevin +// SPDX - License - Identifier: GPL - 3.0 + +#ifndef MANTID_GEOMETRY_INFOITERATORBASE_H_ +#define MANTID_GEOMETRY_INFOITERATORBASE_H_ + +#include <algorithm> +#include <boost/iterator/iterator_facade.hpp> + +namespace Mantid { +namespace Geometry { + +/** InfoIterator + +Base to allow users of the Info objects (DetectorInfo etc) access to data +via a random access iterator. + +Note that the reference type (InfoItem<T>) causes the iterator to be treated as +std::input_iterator for the purposes of many std algorithms such as +std::advance. See https://en.cppreference.com/w/cpp/iterator/advance for example +*/ +template <typename T, template <typename> class InfoItem> +class InfoIteratorBase + : public boost::iterator_facade<InfoIteratorBase<T, InfoItem>, InfoItem<T>, + boost::random_access_traversal_tag, + InfoItem<T>> { + +public: + /** + * Constructor for base iterator + * @param info : Info object (T) to provide iterator ontop of. + * @param index : start point of iterator + * @param totalSize : Represents maximum length of info. i.e. total number of + * items that can be iterated over. + */ + InfoIteratorBase(T &info, const size_t index, const size_t totalSize) + : m_item(info, index), m_totalSize(totalSize) { + if (index > totalSize) + throw std::invalid_argument( + "Iterator start point cannot be greater than maximum size"); + } + +private: + // Allow boost iterator access + friend class boost::iterator_core_access; + + void advance(int64_t delta) { + m_item.m_index = + delta < 0 ? std::max(static_cast<uint64_t>(0), + static_cast<uint64_t>(m_item.m_index) + delta) + : std::min(m_totalSize, + m_item.m_index + static_cast<size_t>(delta)); + } + + bool equal(const InfoIteratorBase<T, InfoItem> &other) const { + return getIndex() == other.getIndex(); + } + + void increment() { + if (m_item.m_index < m_totalSize) { + ++m_item.m_index; + } + } + + void decrement() { + if (m_item.m_index > 0) { + --m_item.m_index; + } + } + + size_t getIndex() const { return m_item.m_index; } + + void setIndex(const size_t index) { m_item.m_index = index; } + + InfoItem<T> dereference() const { return m_item; } + + uint64_t distance_to(const InfoIteratorBase<T, InfoItem> &other) const { + return static_cast<uint64_t>(other.getIndex()) - + static_cast<uint64_t>(getIndex()); + } + + InfoItem<T> m_item; + size_t m_totalSize; +}; +} // namespace Geometry +} // namespace Mantid +#endif /* MANTID_GEOMETRY_INFOITERATORBASE_H_ */ diff --git a/Framework/Geometry/src/Instrument/ComponentInfo.cpp b/Framework/Geometry/src/Instrument/ComponentInfo.cpp index d15a38008cd0c609daeadae94a40d2276683bb66..c7506abf23c33126be0e6da6925a033d5b0fac8a 100644 --- a/Framework/Geometry/src/Instrument/ComponentInfo.cpp +++ b/Framework/Geometry/src/Instrument/ComponentInfo.cpp @@ -440,5 +440,21 @@ void ComponentInfo::merge(const ComponentInfo &other) { m_componentInfo->merge(*other.m_componentInfo); } +ComponentInfoIt ComponentInfo::begin() { + return ComponentInfoIt(*this, 0, size()); +} + +ComponentInfoIt ComponentInfo::end() { + return ComponentInfoIt(*this, size(), size()); +} + +const ComponentInfoConstIt ComponentInfo::cbegin() { + return ComponentInfoConstIt(*this, 0, size()); +} + +const ComponentInfoConstIt ComponentInfo::cend() { + return ComponentInfoConstIt(*this, size(), size()); +} + } // namespace Geometry } // namespace Mantid diff --git a/Framework/Geometry/src/Instrument/ComponentInfoIterator.cpp b/Framework/Geometry/src/Instrument/ComponentInfoIterator.cpp new file mode 100644 index 0000000000000000000000000000000000000000..6f836773d1a5ecea63d909ff291c8e7b872c0006 --- /dev/null +++ b/Framework/Geometry/src/Instrument/ComponentInfoIterator.cpp @@ -0,0 +1,5 @@ +#include "MantidGeometry/Instrument/ComponentInfoIterator.h" + +namespace Mantid { +namespace Geometry {} // namespace Geometry +} // namespace Mantid diff --git a/Framework/Geometry/src/Instrument/DetectorInfo.cpp b/Framework/Geometry/src/Instrument/DetectorInfo.cpp index 5f8d6656a1816f61b8f526c02ff6807c87e8a67a..d09e3c348a6001232aa0ecd68ae9aa448004ae16 100644 --- a/Framework/Geometry/src/Instrument/DetectorInfo.cpp +++ b/Framework/Geometry/src/Instrument/DetectorInfo.cpp @@ -332,11 +332,11 @@ DetectorInfo::scanIntervals() const { } const DetectorInfoConstIt DetectorInfo::cbegin() const { - return DetectorInfoConstIt(*this, 0); + return DetectorInfoConstIt(*this, 0, size()); } const DetectorInfoConstIt DetectorInfo::cend() const { - return DetectorInfoConstIt(*this, size()); + return DetectorInfoConstIt(*this, size(), size()); } const Geometry::IDetector &DetectorInfo::getDetector(const size_t index) const { @@ -358,10 +358,14 @@ DetectorInfo::getDetectorPtr(const size_t index) const { } // Begin method for iterator -DetectorInfoIt DetectorInfo::begin() { return DetectorInfoIt(*this, 0); } +DetectorInfoIt DetectorInfo::begin() { + return DetectorInfoIt(*this, 0, size()); +} // End method for iterator -DetectorInfoIt DetectorInfo::end() { return DetectorInfoIt(*this, size()); } +DetectorInfoIt DetectorInfo::end() { + return DetectorInfoIt(*this, size(), size()); +} } // namespace Geometry } // namespace Mantid diff --git a/Framework/Geometry/test/ComponentInfoIteratorTest.h b/Framework/Geometry/test/ComponentInfoIteratorTest.h new file mode 100644 index 0000000000000000000000000000000000000000..f978dac916f5ee1692e8755a9127d48f09fe1e00 --- /dev/null +++ b/Framework/Geometry/test/ComponentInfoIteratorTest.h @@ -0,0 +1,89 @@ +// Mantid Repository : https://github.com/mantidproject/mantid +// +// Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI, +// NScD Oak Ridge National Laboratory, European Spallation Source +// & Institut Laue - Langevin +// SPDX - License - Identifier: GPL - 3.0 + +#ifndef MANTID_GEOMETRY_COMPONENTINFOITERATORTEST_H_ +#define MANTID_GEOMETRY_COMPONENTINFOITERATORTEST_H_ + +#include <cxxtest/TestSuite.h> + +#include "MantidGeometry/Instrument/ComponentInfo.h" +#include "MantidGeometry/Instrument/ComponentInfoItem.h" +#include "MantidGeometry/Instrument/ComponentInfoIterator.h" +#include "MantidGeometry/Instrument/Detector.h" +#include "MantidGeometry/Instrument/DetectorInfo.h" +#include "MantidGeometry/Instrument/InstrumentVisitor.h" +#include "MantidTestHelpers/ComponentCreationHelper.h" +#include <iterator> + +using namespace ComponentCreationHelper; +using namespace Mantid::Geometry; + +class ComponentInfoIteratorTest : public CxxTest::TestSuite { +public: + // This pair of boilerplate methods prevent the suite being created statically + // This means the constructor isn't called when running other tests + static ComponentInfoIteratorTest *createSuite() { + return new ComponentInfoIteratorTest(); + } + static void destroySuite(ComponentInfoIteratorTest *suite) { delete suite; } + + std::unique_ptr<Mantid::Geometry::ComponentInfo> + create_component_info_object() { + + // Create a very basic instrument to visit + auto visitee = createMinimalInstrument(V3D(0, 0, 0), // Source position + V3D(10, 0, 0), // Sample position + V3D(11, 0, 0)); // Detector position + + // Create the instrument visitor + InstrumentVisitor visitor(visitee); + + // Return the DetectorInfo object + return InstrumentVisitor::makeWrappers(*visitee, nullptr).first; + } + + void test_iterator_cbegin() { + auto componentInfo = create_component_info_object(); + auto iter = componentInfo->cbegin(); + // Check we start at the correct place + TS_ASSERT(iter != componentInfo->cend()); + TS_ASSERT_EQUALS(iter->m_index, 0); + } + void test_iterator_cend() { + auto componentInfo = create_component_info_object(); + auto iter = componentInfo->cend(); + // Check we end at the correct place + TS_ASSERT(iter != componentInfo->cbegin()); + TS_ASSERT_EQUALS(iter->m_index, componentInfo->size()); + } + void test_increment_upwards() { + // Iterator starts at component index 0 (detectors usually) and finishes at + // root. + auto componentInfo = create_component_info_object(); + ComponentInfoConstIt it(*componentInfo, 0, componentInfo->size()); + TS_ASSERT(it->isDetector()); + std::advance(it, componentInfo->size() - 1); + TS_ASSERT(!it->isDetector()); // Root is not a detector + } + + void test_detector_components_behave_as_expected() { + + auto componentInfo = create_component_info_object(); + size_t detectorCount = 0; + for (auto item : *componentInfo) { + if (item.isDetector()) { + ++detectorCount; + TS_ASSERT_EQUALS(item.detectorsInSubtree().size(), 1); // Self + TS_ASSERT_EQUALS(item.componentsInSubtree().size(), 1); // Self + TS_ASSERT_EQUALS(item.children().size(), + 0); // Detectors have no children + } + } + TS_ASSERT_EQUALS(detectorCount, 1); // See instrument description above + } +}; + +#endif /* MANTID_GEOMETRY_COMPONENTINFOITERATORTEST_H_ */ diff --git a/Framework/Geometry/test/DetectorInfoIteratorTest.h b/Framework/Geometry/test/DetectorInfoIteratorTest.h index bcf5aadeaa39594ba2f03c25317cf64ebda3ee8e..4626086642ea537c7a47e42aa1102194d2d80123 100644 --- a/Framework/Geometry/test/DetectorInfoIteratorTest.h +++ b/Framework/Geometry/test/DetectorInfoIteratorTest.h @@ -14,8 +14,10 @@ #include "MantidGeometry/Instrument/DetectorInfoIterator.h" #include "MantidGeometry/Instrument/InstrumentVisitor.h" #include "MantidTestHelpers/ComponentCreationHelper.h" - #include <cxxtest/TestSuite.h> +#include <iterator> +#include <type_traits> +#include <typeinfo> using namespace ComponentCreationHelper; using namespace Mantid::Geometry; @@ -146,6 +148,26 @@ public: TS_ASSERT(iter == detectorInfo->cbegin()); } + void test_iterator_catagory() { + // Characterisation tests + using ItTag = + typename std::iterator_traits<DetectorInfoConstIt>::iterator_category; + using InputItTag = std::input_iterator_tag; + using BidirectionalItTag = std::bidirectional_iterator_tag; + const static bool inputit = std::is_convertible<ItTag, InputItTag>::value; + const static bool bidirectionalit = + std::is_convertible<ItTag, BidirectionalItTag>::value; + TSM_ASSERT("Iterator expected to be treated as input_iterator", inputit); + // Assert below. Iterator not bidirectional. This is why decrement via + // std::advance is not supported. Iterator reference must be true reference + // to support this. + TSM_ASSERT( + "Iterator expected not to be treated as legacy bidirectional iterator", + !bidirectionalit); + + // see https://en.cppreference.com/w/cpp/iterator/advance + } + void test_iterator_advance_and_positions() { // Get the DetectorInfo object auto detectorInfo = create_detector_info_object(); @@ -161,12 +183,12 @@ public: // Go backwards 2 places xValue = 15.0; - std::advance(iter, -2); + iter -= 2; TS_ASSERT_EQUALS(iter->position().X(), xValue) // Go to the start - std::advance(iter, -4); - TS_ASSERT(iter == detectorInfo->cbegin()); + iter -= 4; + TS_ASSERT_EQUALS(iter, detectorInfo->cbegin()); } void test_copy_iterator_and_positions() { diff --git a/Framework/PythonInterface/inc/MantidPythonInterface/api/ComponentInfoPythonIterator.h b/Framework/PythonInterface/inc/MantidPythonInterface/api/ComponentInfoPythonIterator.h new file mode 100644 index 0000000000000000000000000000000000000000..e1cc280bf4e2339b6ee7cb0b9d0af83ca739994c --- /dev/null +++ b/Framework/PythonInterface/inc/MantidPythonInterface/api/ComponentInfoPythonIterator.h @@ -0,0 +1,57 @@ +// Mantid Repository : https://github.com/mantidproject/mantid +// +// Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI, +// NScD Oak Ridge National Laboratory, European Spallation Source +// & Institut Laue - Langevin +// SPDX - License - Identifier: GPL - 3.0 + +#ifndef MANTID_PYTHONINTERFACE_COMPONENTINFOPYTHONITERATOR_H_ +#define MANTID_PYTHONINTERFACE_COMPONENTINFOPYTHONITERATOR_H_ + +#include "MantidGeometry/Instrument/ComponentInfo.h" +#include "MantidGeometry/Instrument/ComponentInfoItem.h" +#include "MantidGeometry/Instrument/ComponentInfoIterator.h" + +#include <boost/python/iterator.hpp> + +using Mantid::Geometry::ComponentInfo; +using Mantid::Geometry::ComponentInfoItem; +using Mantid::Geometry::ComponentInfoIterator; +using namespace boost::python; + +namespace Mantid { +namespace PythonInterface { + +/** ComponentInfoPythonIterator + +ComponentInfoPythonIterator is used to expose ComponentInfoIterator to the +Python side. +*/ + +class ComponentInfoPythonIterator { +public: + explicit ComponentInfoPythonIterator(ComponentInfo &detectorInfo) + : m_begin(detectorInfo.begin()), m_end(detectorInfo.end()), + m_firstOrDone(true) {} + + ComponentInfoItem<ComponentInfo> next() { + if (!m_firstOrDone) + ++m_begin; + else + m_firstOrDone = false; + if (m_begin == m_end) { + m_firstOrDone = true; + objects::stop_iteration_error(); + } + return *m_begin; + } + +private: + ComponentInfoIterator<ComponentInfo> m_begin; + ComponentInfoIterator<ComponentInfo> m_end; + bool m_firstOrDone; +}; + +} // namespace PythonInterface +} // namespace Mantid + +#endif /* MANTID_PYTHONINTERFACE_COMPONENTINFOPYTHONITERATOR_H_ */ diff --git a/Framework/PythonInterface/inc/MantidPythonInterface/api/DetectorInfoPythonIterator.h b/Framework/PythonInterface/inc/MantidPythonInterface/api/DetectorInfoPythonIterator.h index ee194aaaa3ef3a5aa5332263cead2f8c3fe64e1a..42c10470115f524c45e8e5c5b70a513c7dafbdca 100644 --- a/Framework/PythonInterface/inc/MantidPythonInterface/api/DetectorInfoPythonIterator.h +++ b/Framework/PythonInterface/inc/MantidPythonInterface/api/DetectorInfoPythonIterator.h @@ -42,7 +42,7 @@ public: : m_begin(detectorInfo.begin()), m_end(detectorInfo.end()), m_firstOrDone(true) {} - const DetectorInfoItem<DetectorInfo> &next() { + DetectorInfoItem<DetectorInfo> next() { if (!m_firstOrDone) ++m_begin; else diff --git a/Framework/PythonInterface/mantid/api/CMakeLists.txt b/Framework/PythonInterface/mantid/api/CMakeLists.txt index 150e26d0eff5cf17a35ff524553e88461d9b2244..1158ba9e36554d939c067c406c435c0a95102f39 100644 --- a/Framework/PythonInterface/mantid/api/CMakeLists.txt +++ b/Framework/PythonInterface/mantid/api/CMakeLists.txt @@ -118,6 +118,7 @@ set ( INC_FILES ${HEADER_DIR}/api/WorkspacePropertyExporter.h ${HEADER_DIR}/api/SpectrumInfoPythonIterator.h ${HEADER_DIR}/api/DetectorInfoPythonIterator.h + ${HEADER_DIR}/api/ComponentInfoPythonIterator.h ) set ( PY_FILES diff --git a/Framework/PythonInterface/mantid/geometry/CMakeLists.txt b/Framework/PythonInterface/mantid/geometry/CMakeLists.txt index 35bd59965e7a75d9d7752d48eba55b1be29de45b..afb043288b4f5f15a53dd6c39e201b2aa6edd191 100644 --- a/Framework/PythonInterface/mantid/geometry/CMakeLists.txt +++ b/Framework/PythonInterface/mantid/geometry/CMakeLists.txt @@ -41,9 +41,10 @@ set ( EXPORT_FILES src/Exports/ReflectionGenerator.cpp src/Exports/DetectorInfo.cpp src/Exports/DetectorInfoItem.cpp - src/Exports/DetectorInfoIterator.cpp src/Exports/DetectorInfoPythonIterator.cpp src/Exports/ComponentInfo.cpp + src/Exports/ComponentInfoItem.cpp + src/Exports/ComponentInfoPythonIterator.cpp ) ############################################################################################# diff --git a/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfo.cpp b/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfo.cpp index a9b068b3b2e427ea26ecb5857d3562d8d53fe3c4..9ecf5d49229f3a623be9caf3d091ffc866f6dd63 100644 --- a/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfo.cpp +++ b/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfo.cpp @@ -8,6 +8,7 @@ #include "MantidGeometry/Objects/IObject.h" #include "MantidKernel/Quat.h" #include "MantidKernel/V3D.h" +#include "MantidPythonInterface/api/ComponentInfoPythonIterator.h" #include "MantidPythonInterface/core/Converters/WrapWithNDArray.h" #include "MantidPythonInterface/kernel/Policies/VectorToNumpy.h" @@ -19,10 +20,17 @@ using Mantid::Geometry::ComponentInfo; using Mantid::Kernel::Quat; using Mantid::Kernel::V3D; +using Mantid::PythonInterface::ComponentInfoPythonIterator; using namespace Mantid::PythonInterface::Converters; using namespace Mantid::PythonInterface::Policies; using namespace boost::python; +namespace { +ComponentInfoPythonIterator make_pyiterator(ComponentInfo &componentInfo) { + return ComponentInfoPythonIterator(componentInfo); +} +} // namespace + // Function pointers to help resolve ambiguity Mantid::Kernel::V3D (ComponentInfo::*position)(const size_t) const = &ComponentInfo::position; @@ -40,6 +48,8 @@ void (ComponentInfo::*setRotation)(const size_t, const Mantid::Kernel::Quat &) = void export_ComponentInfo() { class_<ComponentInfo, boost::noncopyable>("ComponentInfo", no_init) + .def("__iter__", make_pyiterator) + .def("__len__", &ComponentInfo::size, arg("self"), "Returns the number of components.") @@ -138,5 +148,12 @@ void export_ComponentInfo() { .def("shape", &ComponentInfo::shape, (arg("self"), arg("index")), return_value_policy<reference_existing_object>(), - "Returns the shape of the component identified by 'index'."); + "Returns the shape of the component identified by 'index'.") + + .def("indexOfAny", &ComponentInfo::indexOfAny, (arg("self"), arg("name")), + "Returns the index of any component matching name. Raises " + "ValueError if name not found") + + .def("root", &ComponentInfo::root, arg("self"), + "Returns the index of the root component"); } diff --git a/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfoItem.cpp b/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfoItem.cpp new file mode 100644 index 0000000000000000000000000000000000000000..7e9feba6251bcaf2b671d0689524eed4d0147e36 --- /dev/null +++ b/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfoItem.cpp @@ -0,0 +1,50 @@ +// Mantid Repository : https://github.com/mantidproject/mantid +// +// Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI, +// NScD Oak Ridge National Laboratory, European Spallation Source +// & Institut Laue - Langevin +// SPDX - License - Identifier: GPL - 3.0 + +#include "MantidGeometry/Instrument/ComponentInfoItem.h" +#include "MantidGeometry/Instrument/ComponentInfo.h" +#include "MantidKernel/Quat.h" +#include "MantidKernel/V3D.h" +#include "MantidPythonInterface/core/Converters/WrapWithNDArray.h" +#include "MantidPythonInterface/kernel/Policies/VectorToNumpy.h" +#include <boost/python/class.hpp> +#include <boost/python/make_function.hpp> +#include <boost/python/module.hpp> + +using Mantid::Geometry::ComponentInfo; +using Mantid::Geometry::ComponentInfoItem; +using Mantid::Kernel::V3D; +using namespace boost::python; +using namespace Mantid::PythonInterface::Converters; +using namespace Mantid::PythonInterface::Policies; + +// Export DetectorInfoItem +void export_ComponentInfoItem() { + + // Export to Python + class_<ComponentInfoItem<ComponentInfo>>("ComponentInfoItem", no_init) + .add_property("isDetector", &ComponentInfoItem<ComponentInfo>::isDetector) + .add_property( + "componentsInSubtree", + make_function(&ComponentInfoItem<ComponentInfo>::componentsInSubtree, + return_value_policy<VectorToNumpy>())) + .add_property( + "detectorsInSubtree", + make_function(&ComponentInfoItem<ComponentInfo>::detectorsInSubtree, + return_value_policy<VectorToNumpy>())) + .add_property("position", &ComponentInfoItem<ComponentInfo>::position) + .add_property("rotation", &ComponentInfoItem<ComponentInfo>::rotation) + .add_property("parent", &ComponentInfoItem<ComponentInfo>::parent) + .add_property("hasParent", &ComponentInfoItem<ComponentInfo>::hasParent) + .add_property("scaleFactor", + &ComponentInfoItem<ComponentInfo>::scaleFactor) + .add_property("name", &ComponentInfoItem<ComponentInfo>::name) + .add_property( + "children", + make_function(&ComponentInfoItem<ComponentInfo>::children, + return_value_policy<VectorRefToNumpy<WrapReadOnly>>())) + .add_property("index", &ComponentInfoItem<ComponentInfo>::index); +} diff --git a/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfoPythonIterator.cpp b/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfoPythonIterator.cpp new file mode 100644 index 0000000000000000000000000000000000000000..a07add6f07db2fe2d3be67a1ff3d89cdfe8735b5 --- /dev/null +++ b/Framework/PythonInterface/mantid/geometry/src/Exports/ComponentInfoPythonIterator.cpp @@ -0,0 +1,28 @@ +// Mantid Repository : https://github.com/mantidproject/mantid +// +// Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI, +// NScD Oak Ridge National Laboratory, European Spallation Source +// & Institut Laue - Langevin +// SPDX - License - Identifier: GPL - 3.0 + +#include "MantidPythonInterface/api/ComponentInfoPythonIterator.h" +#include "MantidPythonInterface/core/VersionCompat.h" +#include <boost/python/class.hpp> +#include <boost/python/iterator.hpp> +#include <boost/python/module.hpp> + +using Mantid::PythonInterface::ComponentInfoPythonIterator; +using namespace boost::python; + +// Export ComponentInfoPythonIterator +void export_ComponentInfoPythonIterator() { + + // Export to Python + class_<ComponentInfoPythonIterator>("ComponentInfoPythonIterator", no_init) + .def("__iter__", objects::identity_function()) +#ifdef IS_PY3K + .def("__next__", &ComponentInfoPythonIterator::next) +#else + .def("next", &ComponentInfoPythonIterator::next) +#endif + ; +} diff --git a/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfo.cpp b/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfo.cpp index f348ac26520887c9dfb1adfeeaa6fc775b4fba08..d8e57ebe670007cc40614fa290b53457a2a40c84 100644 --- a/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfo.cpp +++ b/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfo.cpp @@ -62,6 +62,8 @@ void export_DetectorInfo() { void (DetectorInfo::*setMasked)(const size_t, bool) = &DetectorInfo::setMasked; + double (DetectorInfo::*l2)(const size_t) const = &DetectorInfo::l2; + // Export to Python class_<DetectorInfo, boost::noncopyable>("DetectorInfo", no_init) @@ -105,5 +107,9 @@ void export_DetectorInfo() { "Returns the absolute rotation of the detector where the detector " "is identified by 'index'.") .def("detectorIDs", &DetectorInfo::detectorIDs, return_readonly_numpy(), - "Returns all detector ids sorted by detector index"); + "Returns all detector ids sorted by detector index") + .def("l2", l2, (arg("self"), arg("index")), + "Returns the l2 scattering distance") + .def("l1", &DetectorInfo::l1, arg("self"), + "Returns the l1 scattering distance"); } diff --git a/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoItem.cpp b/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoItem.cpp index 09445dbb303b113314cbe711cb0ce91842e914d3..e9b2a2d346105485f89134f4bb554b0ae5c5a4c9 100644 --- a/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoItem.cpp +++ b/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoItem.cpp @@ -27,6 +27,8 @@ void export_DetectorInfoItem() { .add_property("twoTheta", &DetectorInfoItem<DetectorInfo>::twoTheta) .add_property("position", &DetectorInfoItem<DetectorInfo>::position) .add_property("rotation", &DetectorInfoItem<DetectorInfo>::rotation) + .add_property("l2", &DetectorInfoItem<DetectorInfo>::l2) + .add_property("index", &DetectorInfoItem<DetectorInfo>::index) .def("setMasked", &DetectorInfoItem<DetectorInfo>::setMasked, (arg("self"), arg("masked")), "Set the mask flag for the detector"); } diff --git a/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoIterator.cpp b/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoIterator.cpp deleted file mode 100644 index 9f099aff92a1c60986ba11c48adce3fca83d8d2f..0000000000000000000000000000000000000000 --- a/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoIterator.cpp +++ /dev/null @@ -1,22 +0,0 @@ -// Mantid Repository : https://github.com/mantidproject/mantid -// -// Copyright © 2018 ISIS Rutherford Appleton Laboratory UKRI, -// NScD Oak Ridge National Laboratory, European Spallation Source -// & Institut Laue - Langevin -// SPDX - License - Identifier: GPL - 3.0 + -#include "MantidGeometry/Instrument/DetectorInfoIterator.h" -#include "MantidGeometry/Instrument/DetectorInfo.h" - -#include <boost/python/class.hpp> -#include <boost/python/module.hpp> - -using Mantid::Geometry::DetectorInfo; -using Mantid::Geometry::DetectorInfoIterator; -using namespace boost::python; - -// Export DetectorInfoIterator -void export_DetectorInfoIterator() { - - // Export to Python - class_<DetectorInfoIterator<DetectorInfo>>("DetectorInfoIterator", no_init); -} diff --git a/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoPythonIterator.cpp b/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoPythonIterator.cpp index 2f104addd520e52c1b02cd755003388571e59bf8..0c7db21d73591acce9aea5acdf57eb8a0651edef 100644 --- a/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoPythonIterator.cpp +++ b/Framework/PythonInterface/mantid/geometry/src/Exports/DetectorInfoPythonIterator.cpp @@ -5,9 +5,8 @@ // & Institut Laue - Langevin // SPDX - License - Identifier: GPL - 3.0 + #include "MantidPythonInterface/api/DetectorInfoPythonIterator.h" - +#include "MantidPythonInterface/core/VersionCompat.h" #include <boost/python/class.hpp> -#include <boost/python/copy_const_reference.hpp> #include <boost/python/iterator.hpp> #include <boost/python/module.hpp> @@ -20,22 +19,10 @@ void export_DetectorInfoPythonIterator() { // Export to Python class_<DetectorInfoPythonIterator>("DetectorInfoPythonIterator", no_init) .def("__iter__", objects::identity_function()) - .def( -#if PY_VERSION_HEX >= 0x03000000 - "__next__" +#ifdef IS_PY3K + .def("__next__", &DetectorInfoPythonIterator::next) #else - "next" + .def("next", &DetectorInfoPythonIterator::next) #endif - , - &DetectorInfoPythonIterator::next, - return_value_policy<copy_const_reference>()); - /* - Return value policy for next is to copy the const reference. Copy by value is - essential for python 2.0 compatibility because items (DetectorInfoItem) will - outlive their iterators if declared as part of for loops. There is no good - way to deal with this other than to force a copy so that internals of the - item are not also corrupted. Future developers may wish to choose a separte - policy for python 3.0 where this is not a concern, and const ref returns - would be faster. - */ + ; } diff --git a/Framework/PythonInterface/test/python/mantid/geometry/ComponentInfoTest.py b/Framework/PythonInterface/test/python/mantid/geometry/ComponentInfoTest.py index a3de637b28bed02faa035813f5bb0a4fc450f8ce..6ac48ff2ca9019af362b1cf84c3b9d275cb93fbb 100644 --- a/Framework/PythonInterface/test/python/mantid/geometry/ComponentInfoTest.py +++ b/Framework/PythonInterface/test/python/mantid/geometry/ComponentInfoTest.py @@ -14,6 +14,7 @@ from mantid.kernel import V3D from mantid.kernel import Quat from mantid.geometry import CSGObject from mantid.simpleapi import * +from itertools import islice class ComponentInfoTest(unittest.TestCase): @@ -185,7 +186,16 @@ class ComponentInfoTest(unittest.TestCase): info = workspace.componentInfo() self.assertEquals(info.size(), 1) - + def test_indexOfAny(self): + info = self._ws.componentInfo() + index = info.indexOfAny(info.name(info.root())) + # Root index and the discovered index should be the same + self.assertEquals(index, info.root()) + + def test_indexOfAny_throws(self): + info = self._ws.componentInfo() + with self.assertRaises(ValueError): + info.indexOfAny('fictitious') """ ---------------------------------------------------------------------------- Extreme Tests @@ -474,5 +484,66 @@ class ComponentInfoTest(unittest.TestCase): with self.assertRaises(TypeError): info.shape(11.32) + def test_basic_iterator(self): + info = self._ws.componentInfo() + expected_iterations = len(info) + actual_iterations = len(list(iter(info))) + self.assertEquals(expected_iterations, actual_iterations) + it = iter(info) + self.assertEquals(next(it).index, 0) + self.assertEquals(next(it).index, 1) + + def test_isDetector_via_iterator(self): + comp_info = self._ws.componentInfo() + n_detectors = len(self._ws.detectorInfo()) + it = iter(comp_info) + self.assertEquals(next(it).isDetector, True) + self.assertEquals(next(it).isDetector, True) + self.assertEquals(next(it).isDetector, False) + self.assertEquals(next(it).isDetector, False) + + def test_position_via_iterator(self): + comp_info = self._ws.componentInfo() + source_pos = comp_info.sourcePosition() + it = iter(comp_info) + # basic check on first detector position + self.assertTrue(next(it).position.distance(source_pos) > 0) + + def test_children_via_iterator(self): + info = self._ws.componentInfo() + it = iter(info) + first_det = next(it) + self.assertEquals(type(first_det.children), np.ndarray) + self.assertEquals(len(first_det.children), 0) + root = next(it) + for root in it: + continue + self.assertEquals(root.index, info.root()) # sanity check + self.assertTrue(np.array_equal(root.children, np.array([0,1,2,3,4], dtype='uint64'))) + + def test_detectorsInSubtree_via_iterator(self): + info = self._ws.componentInfo() + it = iter(info) + first_det = next(it) + self.assertEquals(type(first_det.detectorsInSubtree), np.ndarray) + # For detectors, only contain own index + self.assertTrue(np.array_equal(first_det.detectorsInSubtree,np.array([0], dtype='uint64'))) + root = next(it) + for root in it: + continue + self.assertTrue(np.array_equal(root.detectorsInSubtree, np.array([0,1], dtype='uint64'))) + + def test_componentsInSubtree_via_iterator(self): + info = self._ws.componentInfo() + it = iter(info) + first_det = next(it) + self.assertEquals(type(first_det.detectorsInSubtree), np.ndarray) + # For detectors, only contain own index + self.assertTrue(np.array_equal(first_det.componentsInSubtree,np.array([0], dtype='uint64'))) + root = next(it) + for root in it: + continue + # All component indices expected including self + self.assertTrue(np.array_equal(root.componentsInSubtree, np.array([0,1,2,3,4,5], dtype='uint64'))) if __name__ == '__main__': unittest.main() diff --git a/Framework/PythonInterface/test/python/mantid/geometry/DetectorInfoTest.py b/Framework/PythonInterface/test/python/mantid/geometry/DetectorInfoTest.py index a9ca56956b6369c20a6f475597f0fd513c4dd91c..2419c714d0e4a9b2bf33901030542b94d1759203 100644 --- a/Framework/PythonInterface/test/python/mantid/geometry/DetectorInfoTest.py +++ b/Framework/PythonInterface/test/python/mantid/geometry/DetectorInfoTest.py @@ -109,7 +109,19 @@ class DetectorInfoTest(unittest.TestCase): self.assertEquals(type(info.rotation(0)), Quat) self.assertEquals(type(info.rotation(1)), Quat) - + def test_l2(self): + det_info = self._ws.detectorInfo() + sample_pos = self._ws.componentInfo().samplePosition() + l2_calc = det_info.position(0).distance(sample_pos) + self.assertEquals(det_info.l2(0), l2_calc) + + def test_l1(self): + source_pos = self._ws.componentInfo().sourcePosition() + sample_pos = self._ws.componentInfo().samplePosition() + l1_calc = sample_pos.distance(source_pos) + det_info = self._ws.detectorInfo() + self.assertEquals(det_info.l1(), l1_calc) + """ --------------- Iteration @@ -121,6 +133,9 @@ class DetectorInfoTest(unittest.TestCase): expected_iterations = len(info) actual_iterations = len(list(iter(info))) self.assertEquals(expected_iterations, actual_iterations) + it = iter(info) + self.assertEquals(next(it).index, 0) + self.assertEquals(next(it).index, 1) def test_iterator_for_monitors(self): info = self._ws.detectorInfo() @@ -155,6 +170,10 @@ class DetectorInfoTest(unittest.TestCase): self.assertTrue(pos.Y() > lastY) lastY = pos.Y() + def test_iterator_for_l2(self): + info = self._ws.detectorInfo() + for item in info: + self.assertTrue(item.l2 > 0) """ ---------------------------------------------------------------------------- diff --git a/docs/source/concepts/ComponentInfo.rst b/docs/source/concepts/ComponentInfo.rst index 80ce4b8f77f7d0230210e6eb3f9afe5fae3778a6..5aa5257c45e70e8cbddc993c29df10203163e9a6 100644 --- a/docs/source/concepts/ComponentInfo.rst +++ b/docs/source/concepts/ComponentInfo.rst @@ -9,7 +9,26 @@ ComponentInfo Introduction ------------ -``ComponentInfo`` provides faster and simpler access to instrument/beamline geometry as required by Mantid :ref:`Algorithms <Algorithm>` than was possible using :ref:`Instrument`. ``ComponentInfo`` and :ref:`DetectorInfo` are designed as full replacements to :ref:`Instrument`. At the time of writing, `ComponentInfo` is incomplete. +``ComponentInfo`` provides faster and simpler access to instrument/beamline geometry as required by Mantid :ref:`Algorithms <Algorithm>` than was possible using :ref:`Instrument`. ``ComponentInfo`` and :ref:`DetectorInfo` are designed as full replacements to :ref:`Instrument`. -Until complete, :ref:`Instrument Access Layers <InstrumentAccessLayers>` provides the best details on how `ComponentInfo` can be used in it's current state of implementation. +:ref:`Instrument Access Layers <InstrumentAccessLayers>` provides details on how `DetectorInfo` interacts with other geometry access layers. + +Python Interface +---------------- + +Examples of using ``ComponentInfo`` in python + +Print indices of detectors in "bank1" that are masked + +.. code-block:: python + + from mantid.simpleapi import CreateSampleWorkspace + + ws = CreateSampleWorkspace() + comp_info = ws.componentInfo() + det_info = ws.detectorInfo() + bank_index compinfo.indexOfAny('bank1') + for det_index in compinfo.detectorsInSubtree(bank_index): + if det_info.isMasked(int(det_index)): + print('Masked detector of bank1, ', det_index) diff --git a/docs/source/concepts/DetectorInfo.rst b/docs/source/concepts/DetectorInfo.rst index 9af68a98968c52cbe8d17ea303e5fa9c72ada3f9..a7e6783b04c445f992e6565ddc7c6f74a62cd3fb 100644 --- a/docs/source/concepts/DetectorInfo.rst +++ b/docs/source/concepts/DetectorInfo.rst @@ -9,6 +9,37 @@ DetectorInfo Introduction ------------ -``DetectorInfo`` provides faster and simpler access to instrument/beamline detector geometry and metadata as required by Mantid :ref:`Algorithms <Algorithm>` than was possible using :ref:`Instrument`. ``DetectorInfo`` and :ref:`ComponentInfo` are designed as full replacements to :ref:`Instrument`. At the time of writing, `DetectorInfo` is incomplete. +``DetectorInfo`` provides faster and simpler access to instrument/beamline detector geometry and metadata as required by Mantid :ref:`Algorithms <Algorithm>` than was possible using :ref:`Instrument`. ``DetectorInfo`` and :ref:`ComponentInfo` are designed as full replacements to :ref:`Instrument`. -Until complete, :ref:`Instrument Access Layers <InstrumentAccessLayers>` provides the best details on how `DetectorInfo` can be used in it's current state of implementation. +:ref:`Instrument Access Layers <InstrumentAccessLayers>` provides details on how `DetectorInfo` interacts with other geometry access layers. + +Python Interface +---------------- + +Examples of using ``DetectorInfo`` in python + +Mask detectors at some distance from the source + +.. code-block:: python + + from mantid.simpleapi import CreateSampleWorkspace + + # Test workspace with instrument + ws = CreateSampleWorkspace() + det_info = ws.detectorInfo(); + for item in det_info: + if not item.isMonitor and item.l2 > 2.0: + item.setMasked(True) + +Print detectors with scattering angle + +.. code-block:: python + + from mantid.simpleapi import CreateSampleWorkspace + + # Test workspace with instrument + ws = CreateSampleWorkspace() + det_info = ws.detectorInfo() + for item in det_info: + if item.l2 > 0.01: + print(item.index) diff --git a/docs/source/concepts/InstrumentAccessLayers.rst b/docs/source/concepts/InstrumentAccessLayers.rst index 65543664651ed6df60ae362c3e608b141c585e92..7161ca559b38b64f7c4f8b0704ced739cd6f4eac 100644 --- a/docs/source/concepts/InstrumentAccessLayers.rst +++ b/docs/source/concepts/InstrumentAccessLayers.rst @@ -37,12 +37,20 @@ There is also a near-complete implementation of the "real" ``DetectorInfo`` clas ``ExperimentInfo`` now also provides a method ``mutableDetectorInfo()`` so that non-const access to the DetectorInfo is possible for purposes of writing detector related information such as positions or rotations. +The python interface to ``DetectorInfo`` has matured, and includes widespread immutable access via iterators. The iterators can also be used to set masking flags. + +See :ref:`DetectorInfo <DetectorInfo>` for more information. + ComponentInfo ______________ ``ComponentInfo`` can be obtatined from a call to ``ExperimentInfo::componentInfo()`` (usually this method would be called on ``MatrixWorkspace``). Much like ``DetectorInfo``, the ``ComponentInfo`` yielded from this method call is a wrapper, which contains shape and index information, that cannot yet be moved in to the real ``Beamline::ComponentInfo``. However, replacing existing usage of ``IComponent`` and ``IObjComponent`` wherever possible with ``ComponentInfo`` across the framework will represent a major step forwards. For writing to the component tree. You can extract a non-const ``ComponentInfo`` via ``ExperimentInfo::mutableComponentInfo``. +The python interface to ``ComponentInfo`` has matured, and now provides equal, if not better support than the ``Instrument`` API for navigating the high-level instrument. Iterator support has also been provided for ``ComponentInfo``. + +See :ref:`ComponentInfo <ComponentInfo>` for more information. + Changes for Rollout ------------------- diff --git a/docs/source/release/v3.14.0/framework.rst b/docs/source/release/v3.14.0/framework.rst index b307c9daf7d01f1c7f7169f05960c942fb115c87..9aac202867f903faf57ca6d1b3f1b22fe7b4048d 100644 --- a/docs/source/release/v3.14.0/framework.rst +++ b/docs/source/release/v3.14.0/framework.rst @@ -126,8 +126,8 @@ New * :class:`mantid.geometry.DetectorInfo` * :class:`mantid.api.SpectrumInfo` -- :class:`mantid.geometry.ComponentInfo` is exposed to allow the user to access geometric information about the components which are part of a beamline. -- :class:`mantid.geometry.DetectorInfo` offers the user the ability to access geometric information about the detector(s) which are part of a beamline. ``DetectorInfo`` has also been given an iterator to allow users to write more Pythonic loops rather than normal index based loops. +- :class:`mantid.geometry.ComponentInfo` is exposed to allow the user to access geometric information about the components which are part of a beamline. Iterator support is also provided via python. +- :class:`mantid.geometry.DetectorInfo` offers the user the ability to access geometric information about the detector(s) which are part of a beamline. ``DetectorInfo`` has also been given a python iterator. - :class:`mantid.api.SpectrumInfo` allows the user to access information about the spectra being used in a beamline. ``SpectrumInfo`` has also been given an iterator to allow users to write more Pythonic loops rather than normal index based loops. In addition to this ``SpectrumDefinition`` objects can also be accessed via a :class:`mantid.api.SpectrumInfo` object. The ``SpectrumDefinition`` object can be used to obtain information about the spectrum to detector mapping and provides a definition of what a spectrum comprises, i.e. indices of all detectors that contribute to the data stored in the spectrum. - Added new :ref:`unit <Unit Factory>` called ``Temperature`` which has units of Kelvin. - Importing ``mantid`` no longer initializes the ``FrameworkManager``. This allows separate classes to be imported without requiring a long delay in waiting for the framework to start. Amongst other things this allows the application name to be set correctly: