Newer
Older
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
#include "MantidAlgorithms/SANSCollimationLengthEstimator.h"
#include "MantidAPI/MatrixWorkspace.h"
#include "MantidKernel/Logger.h"
#include "MantidKernel/TimeSeriesProperty.h"
#include "MantidKernel/V3D.h"
namespace{
Mantid::Kernel::Logger g_log("SANSCollimationLengthEstimator");
}
namespace Mantid {
namespace Algorithms {
using namespace Kernel;
using namespace API;
SANSCollimationLengthEstimator::SANSCollimationLengthEstimator() {}
SANSCollimationLengthEstimator::~SANSCollimationLengthEstimator() {}
/**
* Provide the collimation length which is associated with the instrument
* @param workspace: the input workspace
* @returns the collimation length
*/
double SANSCollimationLengthEstimator::provideCollimationLength(Mantid::API::MatrixWorkspace_sptr workspace) {
// If the instrument does not have a correction specified then set the length to 4
const double defaultLColim = 4.0;
auto collimationLengthID = "collimation-length-correction";
if (!workspace->getInstrument()->hasParameter(collimationLengthID)) {
g_log.error("Error in SANSCollimtionLengthEstimator: The instrument parameter file does not contain a collimation length correction,"
"a default of 4 is provided. Please update the instrument parameter file.");
return defaultLColim;
}
// Get the L1 length
const V3D samplePos = workspace->getInstrument()->getSample()->getPos();
const V3D sourcePos = workspace->getInstrument()->getSource()->getPos();
const V3D SSD = samplePos - sourcePos;
const double L1 = SSD.norm();
auto collimationLengthCorrection = workspace->getInstrument()->getNumberParameter(collimationLengthID);
if (workspace->getInstrument()->hasParameter("special-default-collimation-length-method")){
auto specialCollimationMethod =workspace->getInstrument()->getStringParameter("special-default-collimation-length-method");
if (specialCollimationMethod[0] == "guide") {
return getCollimationLengthWithGuides(workspace, L1, collimationLengthCorrection[0]);
} else {
throw std::invalid_argument("Error in SANSCollimationLengthEstimator: Unknown special collimation method.");
}
} else {
return L1 - collimationLengthCorrection[0];
}
}
/**
* This extraction strategy gets applied when guides are used to calculate the collimation length. The instrument
* parameter file contains the information about the number of guides to use. The guide data itself is fetched
* from the sample logs.
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
* @param inOutWS: a matrix workspace
* @param L1: the distance between sample and source
* @param collimationLengthCorrection: The correction to get the collimation length
*/
double SANSCollimationLengthEstimator::getCollimationLengthWithGuides(MatrixWorkspace_sptr inOutWS, const double L1, const double collimationLengthCorrection) const {
auto lCollim = L1 - collimationLengthCorrection;
// Make sure we have guide cutoffs
if (!inOutWS->getInstrument()->hasParameter("guide-cutoff")) {
throw std::invalid_argument("TOFSANSResolutionByPixel: Could not get a GuideCutoff from the instrument");
}
// Make sure we have a defined number of guidess
if (!inOutWS->getInstrument()->hasParameter("number-of-guides")) {
throw std::invalid_argument("TOFSANSResolutionByPixel: Could not get the number of guides.");
}
// Make sure we have a guide increment specified
if (!inOutWS->getInstrument()->hasParameter("guide-collimation-length-increment")) {
throw std::invalid_argument("TOFSANSResolutionByPixel: Could not find a guid increment.");
}
auto numberOfGuides = static_cast<unsigned int>(inOutWS->getInstrument()->getNumberParameter("number-of-guides")[0]);
auto guideIncrement = inOutWS->getInstrument()->getNumberParameter("guide-collimation-length-increment")[0];
// Make sure that all guides are there. They are labelled as Guide1, Guide2, Guide3, ...
// The entry is a numeric TimeSeriesProperty
std::vector<double> guideValues;
for (unsigned int i = 1; i <= numberOfGuides; i++) {
auto guideName = "Guide" + boost::lexical_cast<std::string>(i);
if (inOutWS->run().hasProperty(guideName)) {
auto guideProperty = inOutWS->run().getTimeSeriesProperty<double>(guideName);
guideValues.push_back(guideProperty->firstValue());
} else {
throw std::invalid_argument("TOFSANSResolutionByPixel: Mismatch between specified number of Guides and actual Guides.");
}
}
auto guideCutoff = inOutWS->getInstrument()->getNumberParameter("guide-cutoff")[0];
// Go through the guides and check in an alternate manner if the guide is smaller
// or larger than the cut off value. We start at the last guide and check that it is
// larger than the cutoff, the next one has to be smaller and so on. For example in pseudocode
// If Guide5 > 130: LCollim+=2.0 else break;
// If Guide4 < 130: LCollim+=2.0 else break;
// If Guide3 > 130: LCollim+=2.0 else break;
// ...
unsigned int largerSmallerCounter = 0;
for (auto it = guideValues.rbegin(); it != guideValues.rend(); ++it) {
bool guideIsLarger = largerSmallerCounter%2 == 0;
if (guideIsLarger && (*it > guideCutoff)) {
} else if (!guideIsLarger && (*it < guideCutoff)){