28#include "MantidHistogramData/EstimatePolynomial.h"
29#include "MantidHistogramData/Histogram.h"
30#include "MantidHistogramData/HistogramBuilder.h"
31#include "MantidHistogramData/HistogramIterator.h"
39#include "boost/algorithm/string.hpp"
40#include "boost/algorithm/string/trim.hpp"
45using namespace Algorithms::PeakParameterHelper;
51using Mantid::HistogramData::Histogram;
60const std::string START_WKSP_INDEX(
"StartWorkspaceIndex");
61const std::string STOP_WKSP_INDEX(
"StopWorkspaceIndex");
62const std::string PEAK_CENTERS(
"PeakCenters");
63const std::string PEAK_CENTERS_WKSP(
"PeakCentersWorkspace");
64const std::string PEAK_FUNC(
"PeakFunction");
65const std::string BACK_FUNC(
"BackgroundType");
66const std::string FIT_WINDOW_LIST(
"FitWindowBoundaryList");
67const std::string FIT_WINDOW_WKSP(
"FitPeakWindowWorkspace");
68const std::string PEAK_WIDTH_PERCENT(
"PeakWidthPercent");
69const std::string PEAK_PARAM_NAMES(
"PeakParameterNames");
70const std::string PEAK_PARAM_VALUES(
"PeakParameterValues");
71const std::string PEAK_PARAM_TABLE(
"PeakParameterValueTable");
72const std::string FIT_FROM_RIGHT(
"FitFromRight");
73const std::string MINIMIZER(
"Minimizer");
74const std::string COST_FUNC(
"CostFunction");
75const std::string MAX_FIT_ITER(
"MaxFitIterations");
76const std::string BACKGROUND_Z_SCORE(
"FindBackgroundSigma");
77const std::string HIGH_BACKGROUND(
"HighBackground");
78const std::string POSITION_TOL(
"PositionTolerance");
79const std::string PEAK_MIN_HEIGHT(
"MinimumPeakHeight");
80const std::string CONSTRAIN_PEAK_POS(
"ConstrainPeakPositions");
81const std::string COPY_LAST_GOOD_PEAK_PARAMS(
"CopyLastGoodPeakParameters");
82const std::string RESPECT_FIXED_PEAK_PARAMS(
"RespectFixedPeakParameters");
83const std::string OUTPUT_WKSP_MODEL(
"FittedPeaksWorkspace");
84const std::string OUTPUT_WKSP_PARAMS(
"OutputPeakParametersWorkspace");
85const std::string OUTPUT_WKSP_PARAM_ERRS(
"OutputParameterFitErrorsWorkspace");
86const std::string RAW_PARAMS(
"RawPeakParameters");
87const std::string PEAK_MIN_SIGNAL_TO_NOISE_RATIO(
"MinimumSignalToNoiseRatio");
88const std::string PEAK_MIN_TOTAL_COUNT(
"MinimumPeakTotalCount");
89const std::string PEAK_MIN_SIGNAL_TO_SIGMA_RATIO(
"MinimumSignalToSigmaRatio");
93namespace FitPeaksAlgorithm {
99 if (num_peaks == 0 || num_params == 0)
100 throw std::runtime_error(
"No peak or no parameter error.");
104 m_costs.resize(num_peaks, DBL_MAX);
107 for (
size_t ipeak = 0; ipeak < num_peaks; ++ipeak) {
154 throw std::runtime_error(
"Peak index is out of range.");
163 size_t peak_num_params = fit_functions.
peakfunction->nParams();
164 for (
size_t ipar = 0; ipar < peak_num_params; ++ipar) {
169 for (
size_t ipar = 0; ipar < fit_functions.
bkgdfunction->nParams(); ++ipar) {
184 throw std::runtime_error(
"Peak index is out of range");
185 if (peak_position >= 0.)
186 throw std::runtime_error(
"Can only set negative postion for bad record");
234 assert(individual_rejection_count <= 1);
236 return individual_rejection_count == 1;
246 std::ostringstream os;
257 os <<
m_low_snr <<
" peak(s) rejected: low signal-to-noise ratio.\n";
265 : m_fitPeaksFromRight(true), m_fitIterations(50), m_numPeaksToFit(0), m_minPeakHeight(0.),
266 m_minSignalToNoiseRatio(0.), m_minPeakTotalCount(0.), m_peakPosTolCase234(false) {}
273 "Name of the input workspace for peak fitting.");
276 "Name of the output workspace containing peak centers for "
278 "The output workspace is point data."
279 "Each workspace index corresponds to a spectrum. "
280 "Each X value ranges from 0 to N-1, where N is the number of "
282 "Each Y value is the peak position obtained by peak fitting. "
283 "Negative value is used for error signals. "
284 "-1 for data is zero; -2 for maximum value is smaller than "
285 "specified minimum value."
286 "and -3 for non-converged fitting.");
289 auto mustBePositive = std::make_shared<BoundedValidator<int>>();
290 mustBePositive->setLower(0);
291 declareProperty(PropertyNames::START_WKSP_INDEX, 0, mustBePositive,
"Starting workspace index for fit");
293 PropertyNames::STOP_WKSP_INDEX,
EMPTY_INT(),
294 "Last workspace index for fit is the smaller of this value and the workspace index of last spectrum.");
297 "List of peak centers to use as initial guess for fit.");
301 "MatrixWorkspace containing referent peak centers for each spectrum, defined at the same workspace indices.");
303 const std::string peakcentergrp(
"Peak Positions");
308 const std::vector<std::string> peakNames = FunctionFactory::Instance().getFunctionNames<
API::IPeakFunction>();
309 declareProperty(PropertyNames::PEAK_FUNC,
"Gaussian", std::make_shared<StringListValidator>(peakNames),
310 "Use of a BackToBackExponential profile is only reccomended if the "
311 "coeficients to calculate A and B are defined in the instrument "
312 "Parameters.xml file.");
313 const vector<string> bkgdtypes{
"Flat",
"Linear",
"Quadratic"};
314 declareProperty(PropertyNames::BACK_FUNC,
"Linear", std::make_shared<StringListValidator>(bkgdtypes),
315 "Type of Background.");
317 const std::string funcgroup(
"Function Types");
324 "List of boundaries of the peak fitting window corresponding to "
329 "MatrixWorkspace containing peak windows for each peak center in each spectrum, defined at the same "
330 "workspace indices.");
332 auto min = std::make_shared<BoundedValidator<double>>();
336 "The estimated peak width as a "
337 "percentage of the d-spacing "
338 "of the center of the peak. Value must be less than 1.");
340 const std::string fitrangeegrp(
"Peak Range Setup");
347 "List of peak parameters' names");
349 "List of peak parameters' value");
352 "Name of the an optional workspace, whose each column "
353 "corresponds to given peak parameter names, "
354 "and each row corresponds to a subset of spectra.");
356 const std::string startvaluegrp(
"Starting Parameters Setup");
363 "Flag for the order to fit peaks. If true, peaks are fitted "
365 "Otherwise peaks are fitted from leftmost.");
367 const std::vector<std::string> minimizerOptions = API::FuncMinimizerFactory::Instance().getKeys();
370 "Minimizer to use for fitting.");
372 const std::array<string, 2> costFuncOptions = {{
"Least squares",
"Rwp"}};
376 auto min_max_iter = std::make_shared<BoundedValidator<int>>();
377 min_max_iter->setLower(49);
378 declareProperty(PropertyNames::MAX_FIT_ITER, 50, min_max_iter,
"Maximum number of function fitting iterations.");
380 const std::string optimizergrp(
"Optimization Setup");
385 std::ostringstream os;
386 os <<
"Deprecated property. Use " << PropertyNames::PEAK_MIN_SIGNAL_TO_NOISE_RATIO <<
" instead.";
390 "Flag whether the input data has high background compared to peak heights.");
393 "List of tolerance on fitted peak positions against given peak positions."
394 "If there is only one value given, then ");
397 "Used for validating peaks before and after fitting. If a peak's observed/estimated or "
398 "fitted height is under this value, the peak will be marked as error.");
401 "If true peak position will be constrained by estimated positions "
402 "(highest Y value position) and "
403 "the peak width either estimted by observation or calculate.");
406 "If true, initial peak parameters (with the exception of peak centre) "
407 "may be copied from the last successfully fit peak in the spectra.");
410 "If true, peak function parameters that are marked as fixed "
411 "(e.g. parameters calculated from the instrument geometry, such as A and B "
412 "of a BackToBackExponential) remain fixed during fitting. "
413 "If false (default), such parameters are unfixed so they can be refined.");
418 "Name of the output matrix workspace with fitted peak. "
419 "This output workspace has the same dimension as the input workspace."
420 "The Y values belonged to peaks to fit are replaced by fitted value. "
421 "Values of estimated background are used if peak fails to be fit.");
425 "Name of table workspace containing all fitted peak parameters.");
431 "Name of workspace containing all fitted peak parameters' fitting error."
432 "It must be used along with FittedPeaksWorkspace and RawPeakParameters "
436 "false generates table with effective centre/width/height "
437 "parameters. true generates a table with peak function "
441 PropertyNames::PEAK_MIN_SIGNAL_TO_NOISE_RATIO, 0.,
442 "Used for validating peaks before fitting. If the signal-to-noise ratio is under this value, "
443 "the peak will be marked as error. This does not apply to peaks for which the noise cannot be estimated.");
446 "Used for validating peaks before fitting. If the total peak window Y-value count "
447 "is under this value, the peak will be excluded from fitting and calibration.");
450 "Used for validating peaks after fitting. If the signal-to-sigma ratio is under this value, "
451 "the peak will be excluded from fitting and calibration.");
453 const std::string addoutgrp(
"Analysis");
464 map<std::string, std::string> issues;
467 if (!(
isDefault(PropertyNames::START_WKSP_INDEX) &&
isDefault(PropertyNames::STOP_WKSP_INDEX))) {
468 const int startIndex =
getProperty(PropertyNames::START_WKSP_INDEX);
469 const int stopIndex =
getProperty(PropertyNames::STOP_WKSP_INDEX);
470 if (startIndex > stopIndex) {
471 const std::string msg =
472 PropertyNames::START_WKSP_INDEX +
" must be less than or equal to " + PropertyNames::STOP_WKSP_INDEX;
473 issues[PropertyNames::START_WKSP_INDEX] = msg;
474 issues[PropertyNames::STOP_WKSP_INDEX] = msg;
479 bool haveCommonPeakParameters(
false);
480 std::vector<string> suppliedParameterNames =
getProperty(PropertyNames::PEAK_PARAM_NAMES);
481 std::vector<double> peakParamValues =
getProperty(PropertyNames::PEAK_PARAM_VALUES);
482 if ((!suppliedParameterNames.empty()) || (!peakParamValues.empty())) {
483 haveCommonPeakParameters =
true;
484 if (suppliedParameterNames.size() != peakParamValues.size()) {
485 issues[PropertyNames::PEAK_PARAM_NAMES] =
"must have same number of values as PeakParameterValues";
486 issues[PropertyNames::PEAK_PARAM_VALUES] =
"must have same number of values as PeakParameterNames";
491 std::string partablename =
getPropertyValue(PropertyNames::PEAK_PARAM_TABLE);
492 if (!partablename.empty()) {
493 if (haveCommonPeakParameters) {
494 const std::string msg =
"Parameter value table and initial parameter "
495 "name/value vectors cannot be given "
497 issues[PropertyNames::PEAK_PARAM_TABLE] = msg;
498 issues[PropertyNames::PEAK_PARAM_NAMES] = msg;
499 issues[PropertyNames::PEAK_PARAM_VALUES] = msg;
507 if (!suppliedParameterNames.empty()) {
510 std::dynamic_pointer_cast<IPeakFunction>(API::FunctionFactory::Instance().createFunction(peakfunctiontype));
513 std::vector<string> functionParameterNames;
515 functionParameterNames.emplace_back(
m_peakFunction->parameterName(i));
518 const bool failed = std::any_of(suppliedParameterNames.cbegin(), suppliedParameterNames.cend(),
519 [&functionParameterNames](
const auto &parName) {
520 return std::find(functionParameterNames.begin(), functionParameterNames.end(),
521 parName) == functionParameterNames.end();
524 std::string msg =
"Specified invalid parameter for peak function";
525 if (haveCommonPeakParameters)
526 issues[PropertyNames::PEAK_PARAM_NAMES] = msg;
528 issues[PropertyNames::PEAK_PARAM_TABLE] = msg;
533 const std::string error_table_name =
getPropertyValue(PropertyNames::OUTPUT_WKSP_PARAM_ERRS);
534 if (!error_table_name.empty()) {
535 const bool use_raw_params =
getProperty(PropertyNames::RAW_PARAMS);
536 if (!use_raw_params) {
537 issues[PropertyNames::OUTPUT_WKSP_PARAM_ERRS] =
"Cannot be used with " + PropertyNames::RAW_PARAMS +
"=False";
538 issues[PropertyNames::RAW_PARAMS] =
539 "Cannot be False with " + PropertyNames::OUTPUT_WKSP_PARAM_ERRS +
" specified";
578 int start_wi =
getProperty(PropertyNames::START_WKSP_INDEX);
582 int stop_wi =
getProperty(PropertyNames::STOP_WKSP_INDEX);
607 throw std::runtime_error(
"number of peaks to fit is zero.");
613 throw std::runtime_error(
"PeakWidthPercent must be less than 1");
618 double temp =
getProperty(PropertyNames::BACKGROUND_Z_SCORE);
620 std::ostringstream os;
621 os <<
"FitPeaks property \"" << PropertyNames::BACKGROUND_Z_SCORE <<
"\" is deprecated and will be ignored."
653 std::dynamic_pointer_cast<IPeakFunction>(API::FunctionFactory::Instance().createFunction(peakfunctiontype));
657 std::string bkgdname;
658 if (bkgdfunctiontype ==
"Linear")
659 bkgdname =
"LinearBackground";
660 else if (bkgdfunctiontype ==
"Flat") {
661 g_log.
warning(
"There may be problems with Flat background");
662 bkgdname =
"FlatBackground";
664 bkgdname = bkgdfunctiontype;
666 std::dynamic_pointer_cast<IBackgroundFunction>(API::FunctionFactory::Instance().createFunction(bkgdname));
669 API::FunctionFactory::Instance().createFunction(
"LinearBackground"));
675 std::string partablename =
getPropertyValue(PropertyNames::PEAK_PARAM_TABLE);
688 }
else if (peakfunctiontype !=
"Gaussian") {
690 g_log.
warning(
"Neither parameter value table nor initial "
691 "parameter name/value vectors is specified. Fitting might "
692 "not be reliable for peak profile other than Gaussian");
704 std::vector<double> peakwindow =
getProperty(PropertyNames::FIT_WINDOW_LIST);
705 std::string peakwindowname =
getPropertyValue(PropertyNames::FIT_WINDOW_WKSP);
710 if ((!peakwindow.empty()) && peakwindowname.empty()) {
715 throw std::invalid_argument(
716 "Specifying peak windows with a list requires also specifying peak positions with a list.");
719 throw std::invalid_argument(
"Peak window vector must be twice as large as number of peaks.");
724 std::vector<double> peakranges(2);
725 peakranges[0] = peakwindow[i * 2];
726 peakranges[1] = peakwindow[i * 2 + 1];
733 std::stringstream errss;
734 errss <<
"Peak " << i <<
": user specifies an invalid range and peak center against " << peakranges[0] <<
" < "
736 throw std::invalid_argument(errss.str());
739 m_getPeakFitWindow = [
this](std::size_t wi, std::size_t ipeak) -> std::pair<double, double> {
748 }
else if (peakwindow.empty() && peakwindowws !=
nullptr) {
756 if (peakWindowX.empty()) {
757 std::stringstream errss;
758 errss <<
"Peak window required at workspace index " << wi <<
" "
759 <<
"which is undefined in the peak window workspace. "
760 <<
"Ensure workspace indices correspond in peak window workspace and input workspace "
761 <<
"when using start and stop indices.";
762 throw std::invalid_argument(errss.str());
765 if (peakWindowX.size() % 2 != 0) {
766 throw std::invalid_argument(
"The peak window vector must be even, with two edges for each peak center.");
768 if (peakWindowX.size() != peakCenterX.size() * 2) {
769 std::stringstream errss;
770 errss <<
"Peak window workspace index " << wi <<
" has incompatible number of fit windows "
771 << peakWindowX.size() / 2 <<
" with the number of peaks " << peakCenterX.size() <<
" to fit.";
772 throw std::invalid_argument(errss.str());
775 for (
size_t ipeak = 0; ipeak < peakCenterX.size(); ++ipeak) {
776 double left_w_bound = peakWindowX[ipeak * 2];
777 double right_w_bound = peakWindowX[ipeak * 2 + 1];
778 double center = peakCenterX[ipeak];
780 if (!(left_w_bound < center && center < right_w_bound)) {
781 std::stringstream errss;
782 errss <<
"Workspace index " << wi <<
" has incompatible peak window "
783 <<
"(" << left_w_bound <<
", " << right_w_bound <<
") "
784 <<
"with " << ipeak <<
"-th expected peak's center " << center;
785 throw std::runtime_error(errss.str());
789 m_getPeakFitWindow = [
this](std::size_t wi, std::size_t ipeak) -> std::pair<double, double> {
798 }
else if (peakwindow.empty()) {
804 m_getPeakFitWindow = [
this](std::size_t wi, std::size_t ipeak) -> std::pair<double, double> {
813 double left = peak_pos - estimate_peak_width * THREE;
814 double right = peak_pos + estimate_peak_width * THREE;
819 throw std::invalid_argument(
"Without definition of peak window, the "
820 "input workspace must be in unit of dSpacing "
821 "and Delta(D)/D must be given!");
825 throw std::invalid_argument(
"One and only one of peak window array and "
826 "peak window workspace can be specified.");
846 g_log.
notice(
"Peak centers are not specified by peak center workspace");
848 std::string peakpswsname =
getPropertyValue(PropertyNames::PEAK_CENTERS_WKSP);
858 }
else if (
m_peakCenters.empty() && peakcenterws !=
nullptr) {
868 std::stringstream errss;
871 <<
"However, the peak center workspace does not have values defined "
872 <<
"at workspace index " << wi <<
". "
873 <<
"Make sure the workspace indices between input and peak center workspaces correspond.";
875 throw std::invalid_argument(errss.str());
885 std::stringstream errss;
886 errss <<
"One and only one in 'PeakCenters' (vector) and "
887 "'PeakCentersWorkspace' shall be given. "
888 <<
"'PeakCenters' has size " <<
m_peakCenters.size() <<
", and name of peak center workspace "
889 <<
"is " << peakpswsname;
890 throw std::invalid_argument(errss.str());
905 throw std::runtime_error(
"ProcessInputPeakTolerance() must be called after "
906 "ProcessInputPeakCenters()");
923 throw std::runtime_error(
"Number of peak position tolerances and number of "
924 "peaks to fit are inconsistent.");
962 std::map<std::string, size_t> parname_index_map;
963 for (
size_t iparam = 0; iparam <
m_peakFunction->nParams(); ++iparam)
964 parname_index_map.insert(std::make_pair(
m_peakFunction->parameterName(iparam), iparam));
972 auto locator = parname_index_map.find(paramName);
973 if (locator != parname_index_map.end()) {
979 <<
" is not an allowed parameter of peak "
997 std::vector<std::shared_ptr<FitPeaksAlgorithm::PeakFitResult>> fit_result_vector(
m_numSpectraToFit);
999 const int nThreads = FrameworkManager::Instance().getNumOMPThreads();
1002 std::shared_ptr<FitPeaksAlgorithm::PeakFitPreCheckResult> pre_check_result =
1003 std::make_shared<FitPeaksAlgorithm::PeakFitPreCheckResult>();
1005 PRAGMA_OMP(parallel
for schedule(dynamic, 1) )
1006 for (
int ithread = 0; ithread < nThreads; ithread++) {
1012 std::vector<std::vector<double>> lastGoodPeakParameters(
m_numPeaksToFit,
1017 for (
auto wi = iws_begin; wi < iws_end; ++wi) {
1023 std::shared_ptr<FitPeaksAlgorithm::PeakFitResult> fit_result =
1024 std::make_shared<FitPeaksAlgorithm::PeakFitResult>(
m_numPeaksToFit, numfuncparams);
1026 std::shared_ptr<FitPeaksAlgorithm::PeakFitPreCheckResult> spectrum_pre_check_result =
1027 std::make_shared<FitPeaksAlgorithm::PeakFitPreCheckResult>();
1029 fitSpectrumPeaks(
static_cast<size_t>(wi), expected_peak_centers, fit_result, lastGoodPeakParameters,
1030 lastGoodPeakSpectra, spectrum_pre_check_result);
1033 writeFitResult(
static_cast<size_t>(wi), expected_peak_centers, fit_result);
1035 *pre_check_result += *spectrum_pre_check_result;
1043 return fit_result_vector;
1048bool estimateBackgroundParameters(
const Histogram &histogram,
const std::pair<size_t, size_t> &peak_window,
1051 std::vector<double> &vec_y);
1052template <
typename vector_like>
1053void rangeToIndexBounds(
const vector_like &vecx,
const double range_left,
const double range_right,
size_t &left_index,
1054 size_t &right_index);
1057std::vector<std::string> supported_peak_profiles{
"Gaussian",
"Lorentzian",
"PseudoVoigt",
"Voigt",
1058 "BackToBackExponential"};
1065double estimateBackgroundNoise(
const std::vector<double> &vec_y) {
1067 size_t half_number_of_bkg_datapoints{5};
1068 if (vec_y.size() < 2 * half_number_of_bkg_datapoints + 3 )
1073 std::vector<double> vec_bkg;
1074 vec_bkg.resize(2 * half_number_of_bkg_datapoints);
1075 std::copy(vec_y.begin(), vec_y.begin() + half_number_of_bkg_datapoints, vec_bkg.begin());
1076 std::copy(vec_y.end() - half_number_of_bkg_datapoints, vec_y.end(), vec_bkg.begin() + half_number_of_bkg_datapoints);
1080 std::vector<double> vec_bkg_no_outliers;
1081 vec_bkg_no_outliers.resize(vec_bkg.size());
1082 double zscore_crit = 3.;
1083 for (
size_t ii = 0; ii < vec_bkg.size(); ii++) {
1084 if (zscore_vec[ii] <= zscore_crit)
1085 vec_bkg_no_outliers.push_back(vec_bkg[ii]);
1088 if (vec_bkg_no_outliers.size() < half_number_of_bkg_datapoints)
1092 return intensityStatistics.standard_deviation;
1103template <
typename vector_like>
1104void rangeToIndexBounds(
const vector_like &elems,
const double range_left,
const double range_right,
size_t &left_index,
1105 size_t &right_index) {
1106 const auto left_iter = std::lower_bound(elems.cbegin(), elems.cend(), range_left);
1107 const auto right_iter = std::upper_bound(elems.cbegin(), elems.cend(), range_right);
1109 left_index = std::distance(elems.cbegin(), left_iter);
1110 right_index = std::distance(elems.cbegin(), right_iter);
1111 right_index = std::min(right_index, elems.size() - 1);
1121 std::vector<double> &vec_y) {
1125 bkgd_func->function(vectorx, vector_bkgd);
1128 for (
size_t i = 0; i < vec_y.size(); ++i) {
1129 (vec_y)[i] -= vector_bkgd[i];
1138class LoggingOffsetSentry {
1156 const std::shared_ptr<FitPeaksAlgorithm::PeakFitResult> &fit_result,
1157 std::vector<std::vector<double>> &lastGoodPeakParameters,
1158 std::vector<size_t> &lastGoodPeakSpectra,
1159 const std::shared_ptr<FitPeaksAlgorithm::PeakFitPreCheckResult> &pre_check_result) {
1165 fit_result->setBadRecord(i, -1.);
1166 pre_check_result->setNumberOfSpectrumPeaksWithLowCount(
m_numPeaksToFit);
1175 std::stringstream errss;
1176 errss <<
"The FitPeak algorithm requires the CurveFitting library";
1178 throw std::runtime_error(errss.str());
1185 peak_fitter->setProperty(
"Minimizer",
m_minimizer);
1187 peak_fitter->setProperty(
"CalcErrors",
true);
1194 bool neighborPeakSameSpectrum =
false;
1195 size_t number_of_out_of_range_peaks{0};
1196 for (
size_t fit_index = 0; fit_index <
m_numPeaksToFit; ++fit_index) {
1198 size_t peak_index(fit_index);
1203 for (
size_t i = 0; i < bkgdfunction->nParams(); ++i)
1204 bkgdfunction->setParameter(i, 0.);
1206 double expected_peak_pos = expected_peak_centers[peak_index];
1210 auto peakfunction = std::dynamic_pointer_cast<API::IPeakFunction>(
m_peakFunction->clone());
1211 peakfunction->setCentre(expected_peak_pos);
1214 peakfunction->setMatrixWorkspace(
m_inputMatrixWS, wi, peak_window_i.first, peak_window_i.second);
1216 std::map<size_t, double> keep_values;
1217 for (
size_t ipar = 0; ipar < peakfunction->nParams(); ++ipar) {
1218 if (peakfunction->isFixed(ipar)) {
1222 keep_values[ipar] = peakfunction->getParameter(ipar);
1227 peakfunction->unfix(ipar);
1234 bool samePeakCrossSpectrum = (lastGoodPeakParameters[peak_index].size() >
1235 static_cast<size_t>(std::count_if(lastGoodPeakParameters[peak_index].begin(),
1236 lastGoodPeakParameters[peak_index].end(),
1237 [&](
auto const &val) {
return val <= 1e-10; })));
1242 if (wi > 0 && samePeakCrossSpectrum) {
1243 size_t lastGoodWi = lastGoodPeakSpectra[peak_index];
1244 std::shared_ptr<const Geometry::Detector> pdetector =
1245 std::dynamic_pointer_cast<const Geometry::Detector>(
m_inputMatrixWS->getDetector(lastGoodWi));
1246 std::shared_ptr<const Geometry::Detector> cdetector =
1247 std::dynamic_pointer_cast<const Geometry::Detector>(
m_inputMatrixWS->getDetector(wi));
1250 if (pdetector && cdetector) {
1251 auto prev_id = pdetector->getID();
1252 auto curr_id = cdetector->getID();
1253 if (prev_id + 1 != curr_id)
1254 samePeakCrossSpectrum =
false;
1256 samePeakCrossSpectrum =
false;
1262 samePeakCrossSpectrum =
false;
1264 }
catch (
const std::runtime_error &) {
1267 samePeakCrossSpectrum =
false;
1271 if (samePeakCrossSpectrum) {
1273 for (
size_t i = 0; i < peakfunction->nParams(); ++i) {
1274 peakfunction->setParameter(i, lastGoodPeakParameters[peak_index][i]);
1278 for (
size_t i = 0; i < peakfunction->nParams(); ++i) {
1279 peakfunction->setParameter(i, lastGoodPeakParameters[prev_peak_index][i]);
1284 peakfunction->setCentre(expected_peak_pos);
1287 for (
const auto &[ipar,
value] : keep_values) {
1288 peakfunction->setParameter(ipar,
value);
1291 double cost(DBL_MAX);
1292 if (expected_peak_pos <= x0 || expected_peak_pos >= xf) {
1294 peakfunction->setIntensity(0);
1295 number_of_out_of_range_peaks++;
1305 auto useUserSpecifedIfGiven =
1310 g_log.
warning(
"Peak width can be estimated as ZERO. The result can be wrong");
1314 std::shared_ptr<FitPeaksAlgorithm::PeakFitPreCheckResult> peak_pre_check_result =
1315 std::make_shared<FitPeaksAlgorithm::PeakFitPreCheckResult>();
1316 cost =
fitIndividualPeak(wi, peak_fitter, expected_peak_pos, peak_window_i, observe_peak_width, peakfunction,
1317 bkgdfunction, peak_pre_check_result);
1318 if (peak_pre_check_result->isIndividualPeakRejected())
1319 fit_result->setBadRecord(peak_index, -1.);
1323 fit_result->setBadRecord(peak_index, -1.);
1328 *pre_check_result += *peak_pre_check_result;
1330 pre_check_result->setNumberOfOutOfRangePeaks(number_of_out_of_range_peaks);
1342 neighborPeakSameSpectrum =
true;
1343 prev_peak_index = peak_index;
1345 for (
size_t i = 0; i < lastGoodPeakParameters[peak_index].size(); ++i) {
1346 lastGoodPeakParameters[peak_index][i] = peakfunction->getParameter(i);
1348 lastGoodPeakSpectra[peak_index] = wi;
1372 if (firstPeakInSpectrum) {
1378 peak_function->setParameter(param_index, param_value);
1386 observe_peak_shape =
true;
1389 return observe_peak_shape;
1405 const std::vector<double> &expected_peak_positions,
1407 const std::shared_ptr<FitPeaksAlgorithm::PeakFitResult> &fit_result) {
1409 double postol(DBL_MAX);
1423 throw std::runtime_error(
"Peak tolerance out of index");
1431 bool good_fit(
false);
1432 if ((cost < 0) || (cost >= DBL_MAX - 1.) || std::isnan(cost)) {
1438 }
else if (case23) {
1441 if (fitwindow.first < fitwindow.second) {
1444 if (peak_pos < fitwindow.first || peak_pos > fitwindow.second) {
1447 g_log.
debug() <<
"Peak position " << peak_pos <<
" is out of fit "
1448 <<
"window boundary " << fitwindow.first <<
", " << fitwindow.second <<
"\n";
1449 }
else if (peak_fwhm > (fitwindow.second - fitwindow.first)) {
1452 g_log.
debug() <<
"Peak position " << peak_pos <<
" has fwhm "
1453 <<
"wider than the fit window " << fitwindow.second - fitwindow.first <<
"\n";
1459 double left_bound(-1);
1461 left_bound = 0.5 * (expected_peak_positions[peakindex] - expected_peak_positions[peakindex - 1]);
1462 double right_bound(-1);
1464 right_bound = 0.5 * (expected_peak_positions[peakindex + 1] - expected_peak_positions[peakindex]);
1466 left_bound = right_bound;
1467 if (right_bound < left_bound)
1468 right_bound = left_bound;
1469 if (left_bound < 0 || right_bound < 0)
1470 throw std::runtime_error(
"Code logic error such that left or right "
1471 "boundary of peak position is negative.");
1472 if (peak_pos < left_bound || peak_pos > right_bound) {
1474 }
else if (peak_fwhm > (right_bound - left_bound)) {
1477 g_log.
debug() <<
"Peak position " << peak_pos <<
" has fwhm "
1478 <<
"wider than the fit window " << right_bound - left_bound <<
"\n";
1483 }
else if (
fabs(fitfunction.
peakfunction->centre() - expected_peak_positions[peakindex]) > postol) {
1487 <<
fabs(fitfunction.
peakfunction->centre() - expected_peak_positions[peakindex])
1488 <<
" is out of range of tolerance: " << postol <<
"\n";
1495 double adjust_cost(cost);
1498 adjust_cost = DBL_MAX;
1502 if (adjust_cost > DBL_MAX - 1) {
1507 fit_result->setRecord(peakindex, adjust_cost, peak_pos, fitfunction);
1521 throw std::runtime_error(
"No parameters");
1531 std::vector<std::vector<IBackgroundFunction_sptr>> bkgd_functions(
1536 const std::shared_ptr<FitPeaksAlgorithm::PeakFitResult> &fit_result_i = fit_results[output_iws];
1539 throw std::runtime_error(
"There is something wroing with PeakFitResult vector!");
1542 const double chi2 = fit_result_i->getCost(ipeak);
1549 for (
size_t iparam = 0; iparam < num_peakfunc_params; ++iparam)
1550 peak_function->setParameter(iparam, fit_result_i->getParameterValue(ipeak, iparam));
1551 for (
size_t iparam = 0; iparam < num_bkgdfunc_params; ++iparam)
1552 bkgd_function->setParameter(iparam, fit_result_i->getParameterValue(ipeak, num_peakfunc_params + iparam));
1555 peak_function->setMatrixWorkspace(
m_inputMatrixWS, iws, peakwindow.first, peakwindow.second);
1557 peak_functions[output_iws][ipeak] = std::move(peak_function);
1558 bkgd_functions[output_iws][ipeak] = std::move(bkgd_function);
1565 const std::size_t iws =
static_cast<std::size_t
>(iiws);
1569 if (!peak_functions[output_iws][ipeak] || !bkgd_functions[output_iws][ipeak])
1576 auto start_x_iter = std::lower_bound(vec_x.begin(), vec_x.end(), peakwindow.first);
1577 auto stop_x_iter = std::lower_bound(vec_x.begin(), vec_x.end(), peakwindow.second);
1579 if (start_x_iter == stop_x_iter)
1580 throw std::runtime_error(
"Range size is zero in calculateFittedPeaks");
1585 comp_func->addFunction(std::move(peak_functions[output_iws][ipeak]));
1586 comp_func->addFunction(std::move(bkgd_functions[output_iws][ipeak]));
1587 comp_func->function(domain, values);
1590 std::size_t istart =
static_cast<size_t>(start_x_iter - vec_x.begin());
1591 std::size_t istop =
static_cast<size_t>(stop_x_iter - vec_x.begin());
1592 for (std::size_t yindex = istart; yindex < istop; ++yindex) {
1606 auto startX = std::lower_bound(vecX.begin(), vecX.end(), peakWindow.first);
1607 auto stopX = std::lower_bound(vecX.begin(), vecX.end(), peakWindow.second);
1612 peakFunction->function(domain, values);
1613 auto peakValues = values.
toVector();
1616 auto startE = errors.begin() + (startX - vecX.begin());
1617 auto stopE = errors.begin() + (stopX - vecX.begin());
1618 std::vector<double> peakErrors(startE, stopE);
1620 double peakSum = std::accumulate(peakValues.cbegin(), peakValues.cend(), 0.0);
1627bool estimateBackgroundParameters(
const Histogram &histogram,
const std::pair<size_t, size_t> &peak_window,
1631 const auto POLYNOMIAL_ORDER = std::min<size_t>(1, bkgd_function->nParams());
1633 if (peak_window.first >= peak_window.second)
1634 throw std::runtime_error(
"Invalid peak window");
1637 const auto nParams = bkgd_function->nParams();
1638 for (
size_t i = 0; i < nParams; ++i)
1639 bkgd_function->setParameter(i, 0.);
1642 const size_t iback_start = peak_window.first + 10;
1643 const size_t iback_stop = peak_window.second - 10;
1648 if (iback_start < iback_stop) {
1652 double chisq{DBL_MAX};
1653 HistogramData::estimateBackground(POLYNOMIAL_ORDER, histogram, peak_window.first, peak_window.second, iback_start,
1654 iback_stop, bkgd_a0, bkgd_a1, bkgd_a2, chisq);
1656 bkgd_function->setParameter(0, bkgd_a0);
1658 bkgd_function->setParameter(1, bkgd_a1);
1676 return (std::find(supported_peak_profiles.begin(), supported_peak_profiles.end(), peakprofile) !=
1677 supported_peak_profiles.end());
1685 constexpr size_t MIN_POINTS{10};
1689 const auto &points = histogram.points();
1690 size_t start_index =
findXIndex(points.rawData(), fit_window.first);
1691 size_t expected_peak_index =
findXIndex(points.rawData(), expected_peak_pos, start_index);
1692 size_t stop_index =
findXIndex(points.rawData(), fit_window.second, expected_peak_index);
1695 bool good_fit(
false);
1696 if (expected_peak_index - start_index > MIN_POINTS && stop_index - expected_peak_index > MIN_POINTS) {
1699 const std::pair<double, double> vec_min{fit_window.first, points[expected_peak_index + 5]};
1700 const std::pair<double, double> vec_max{points[expected_peak_index - 5], fit_window.second};
1703 for (
size_t n = 0;
n < bkgd_func->nParams(); ++
n)
1704 bkgd_func->setParameter(
n, 0);
1709 if (chi2 < DBL_MAX - 1) {
1717 g_log.
debug() <<
"Don't know what to do with background fitting with single "
1718 <<
"domain function! " << (expected_peak_index - start_index) <<
" points to the left "
1719 << (stop_index - expected_peak_index) <<
" points to the right\n";
1729 const std::pair<double, double> &fitwindow,
const bool estimate_peak_width,
1732 const std::shared_ptr<FitPeaksAlgorithm::PeakFitPreCheckResult> &pre_check_result) {
1733 pre_check_result->setNumberOfSubmittedIndividualPeaks(1);
1734 double cost(DBL_MAX);
1737 size_t min_required_datapoints{peakfunction->nParams() + bkgdfunc->nParams() + 2};
1739 if (number_of_datapoints < min_required_datapoints) {
1740 pre_check_result->setNumberOfPeaksWithNotEnoughDataPoints(1);
1746 pre_check_result->setNumberOfIndividualPeaksWithLowCount(1);
1752 pre_check_result->setNumberOfPeaksWithLowSignalToNoise(1);
1763 estimate_peak_width,
true);
1778 const std::pair<double, double> &peak_range,
const double &expected_peak_center,
1779 bool estimate_peak_width,
bool estimate_background) {
1780 std::stringstream errorid;
1781 errorid <<
"(WorkspaceIndex=" << wsindex <<
" PeakCentre=" << expected_peak_center <<
")";
1784 if (peak_range.first >= peak_range.second) {
1785 std::stringstream msg;
1786 msg <<
"Invalid peak window: xmin>xmax (" << peak_range.first <<
", " << peak_range.second <<
")" << errorid.str();
1787 throw std::runtime_error(msg.str());
1791 const auto &histogram = dataws->histogram(wsindex);
1792 const auto &vector_x = histogram.points();
1793 const auto start_index =
findXIndex(vector_x, peak_range.first);
1794 const auto stop_index =
findXIndex(vector_x, peak_range.second, start_index);
1795 if (start_index == stop_index)
1796 throw std::runtime_error(
"Range size is zero in fitFunctionSD");
1797 std::pair<size_t, size_t> peak_index_window = std::make_pair(start_index, stop_index);
1800 if (estimate_background) {
1801 if (!estimateBackgroundParameters(histogram, peak_index_window, bkgd_function)) {
1807 peak_function->setCentre(expected_peak_center);
1808 int result =
estimatePeakParameters(histogram, peak_index_window, peak_function, bkgd_function, estimate_peak_width,
1811 if (result !=
GOOD) {
1812 peak_function->setCentre(expected_peak_center);
1820 comp_func->addFunction(peak_function);
1821 comp_func->addFunction(bkgd_function);
1822 IFunction_sptr fitfunc = std::dynamic_pointer_cast<IFunction>(comp_func);
1825 fit->setProperty(
"Function", fitfunc);
1826 fit->setProperty(
"InputWorkspace", dataws);
1827 fit->setProperty(
"WorkspaceIndex",
static_cast<int>(wsindex));
1829 fit->setProperty(
"StartX", peak_range.first);
1830 fit->setProperty(
"EndX", peak_range.second);
1831 fit->setProperty(
"IgnoreInvalidData",
true);
1835 double peak_center = peak_function->centre();
1836 double peak_width = peak_function->fwhm();
1837 std::stringstream peak_center_constraint;
1838 peak_center_constraint << (peak_center - 0.5 * peak_width) <<
" < f0." << peak_function->getCentreParameterName()
1839 <<
" < " << (peak_center + 0.5 * peak_width);
1840 fit->setProperty(
"Constraints", peak_center_constraint.str());
1844 g_log.
debug() <<
"[E1201] FitSingleDomain Before fitting, Fit function: " << fit->asString() <<
"\n";
1845 errorid <<
" starting function [" << comp_func->asString() <<
"]";
1848 g_log.
debug() <<
"[E1202] FitSingleDomain After fitting, Fit function: " << fit->asString() <<
"\n";
1850 if (!fit->isExecuted()) {
1851 g_log.
warning() <<
"Fitting peak SD (single domain) failed to execute. " + errorid.str();
1854 }
catch (std::invalid_argument &e) {
1855 errorid <<
": " << e.what();
1861 std::string fitStatus = fit->getProperty(
"OutputStatus");
1862 double chi2{std::numeric_limits<double>::max()};
1863 if (fitStatus ==
"success") {
1864 chi2 = fit->getProperty(
"OutputChi2overDoF");
1872 const size_t wsindex,
const std::pair<double, double> &vec_xmin,
1873 const std::pair<double, double> &vec_xmax) {
1879 std::stringstream errss;
1880 errss <<
"The FitPeak algorithm requires the CurveFitting library";
1881 throw std::runtime_error(errss.str());
1886 fit->setProperty(
"CalcErrors",
true);
1890 std::shared_ptr<MultiDomainFunction> md_function = std::make_shared<MultiDomainFunction>();
1893 md_function->addFunction(std::move(fit_function));
1896 md_function->clearDomainIndices();
1897 md_function->setDomainIndices(0, {0, 1});
1900 fit->setProperty(
"Function", std::dynamic_pointer_cast<IFunction>(md_function));
1901 fit->setProperty(
"InputWorkspace", dataws);
1902 fit->setProperty(
"WorkspaceIndex",
static_cast<int>(wsindex));
1903 fit->setProperty(
"StartX", vec_xmin.first);
1904 fit->setProperty(
"EndX", vec_xmax.first);
1905 fit->setProperty(
"InputWorkspace_1", dataws);
1906 fit->setProperty(
"WorkspaceIndex_1",
static_cast<int>(wsindex));
1907 fit->setProperty(
"StartX_1", vec_xmin.second);
1908 fit->setProperty(
"EndX_1", vec_xmax.second);
1910 fit->setProperty(
"IgnoreInvalidData",
true);
1914 if (!fit->isExecuted()) {
1915 throw runtime_error(
"Fit is not executed on multi-domain function/data. ");
1919 std::string fitStatus = fit->getProperty(
"OutputStatus");
1921 double chi2 = DBL_MAX;
1922 if (fitStatus ==
"success") {
1923 chi2 = fit->getProperty(
"OutputChi2overDoF");
1932 const size_t &ws_index,
const double &expected_peak_center,
1942 fitBackground(ws_index, fit_window, expected_peak_center, high_bkgd_function);
1945 std::vector<double> vec_x, vec_y, vec_e;
1946 getRangeData(ws_index, fit_window, vec_x, vec_y, vec_e);
1949 reduceByBackground(high_bkgd_function, vec_x, vec_y);
1950 for (std::size_t
n = 0;
n < bkgdfunc->nParams(); ++
n)
1951 bkgdfunc->setParameter(
n, 0);
1957 fitFunctionSD(fit, peakfunction, bkgdfunc, reduced_bkgd_ws, 0, {vec_x.front(), vec_x.back()}, expected_peak_center,
1958 observe_peak_shape,
false);
1961 bkgdfunc->setParameter(0, bkgdfunc->getParameter(0) + high_bkgd_function->getParameter(0));
1962 bkgdfunc->setParameter(1, bkgdfunc->getParameter(1) +
1963 high_bkgd_function->getParameter(1));
1966 expected_peak_center,
false,
false);
1974 const std::vector<double> &vec_y,
1975 const std::vector<double> &vec_e) {
1976 std::size_t size = vec_x.size();
1977 std::size_t ysize = vec_y.size();
1979 HistogramBuilder builder;
1981 builder.setY(ysize);
1984 auto &dataX = matrix_ws->mutableX(0);
1985 auto &dataY = matrix_ws->mutableY(0);
1986 auto &dataE = matrix_ws->mutableE(0);
1988 dataX.assign(vec_x.cbegin(), vec_x.cend());
1989 dataY.assign(vec_y.cbegin(), vec_y.cend());
1990 dataE.assign(vec_e.cbegin(), vec_e.cend());
2006 for (std::size_t ipeak = 0; ipeak < expected_position.size(); ++ipeak) {
2022 const std::vector<std::string> ¶m_names,
bool with_chi2) {
2024 table_ws->addColumn(
"int",
"wsindex");
2025 table_ws->addColumn(
"int",
"peakindex");
2026 for (
const auto ¶m_name : param_names)
2027 table_ws->addColumn(
"double", param_name);
2029 table_ws->addColumn(
"double",
"chi2");
2036 newRow << static_cast<int>(iws);
2037 newRow << static_cast<int>(ipeak);
2038 for (
size_t iparam = 0; iparam < numParam; ++iparam)
2059 std::vector<std::string> param_vec;
2063 param_vec.emplace_back(
"centre");
2064 param_vec.emplace_back(
"width");
2065 param_vec.emplace_back(
"height");
2066 param_vec.emplace_back(
"intensity");
2069 for (
size_t iparam = 0; iparam <
m_bkgdFunction->nParams(); ++iparam)
2077 std::string fiterror_table_name =
getPropertyValue(PropertyNames::OUTPUT_WKSP_PARAM_ERRS);
2079 if (fiterror_table_name.empty()) {
2097 std::string fit_ws_name =
getPropertyValue(PropertyNames::OUTPUT_WKSP_MODEL);
2098 if (fit_ws_name.size() == 0) {
2123 g_log.
debug(
"about to calcualte fitted peaks");
2136 const auto &vec_y = histogram.y().rawData();
2137 double total = std::accumulate(vec_y.begin(), vec_y.end(), 0.);
2149 std::vector<double> vec_x, vec_y, vec_e;
2152 double total = std::accumulate(vec_y.begin(), vec_y.end(), 0.);
2163 size_t left_index, right_index;
2165 size_t number_dp = right_index - left_index + 1;
2168 assert(number_dp > 0);
2180 size_t &right_index) {
2182 const auto &orig_x = histogram.x();
2183 rangeToIndexBounds(orig_x, range.first, range.second, left_index, right_index);
2187 if (left_index >= right_index || (
m_inputMatrixWS->isHistogramData() && left_index == right_index - 1)) {
2188 std::stringstream err_ss;
2189 err_ss <<
"Unable to get a valid subset of histogram from given fit window. "
2190 <<
"Histogram X: " << orig_x.front() <<
"," << orig_x.back() <<
"; Range: " << range.first <<
","
2192 throw std::runtime_error(err_ss.str());
2205 std::vector<double> &vec_y, std::vector<double> &vec_e) {
2207 size_t left_index, right_index;
2211 size_t num_elements_x = right_index - left_index;
2213 vec_x.resize(num_elements_x);
2215 const auto &orig_x = histogram.x();
2216 std::copy(orig_x.begin() + left_index, orig_x.begin() + right_index, vec_x.begin());
2218 size_t num_datapoints =
m_inputMatrixWS->isHistogramData() ? num_elements_x - 1 : num_elements_x;
2220 const auto &orig_y = histogram.y().rawData();
2221 const auto &orig_e = histogram.e().rawData();
2222 vec_y.resize(num_datapoints);
2223 vec_e.resize(num_datapoints);
2224 std::copy(orig_y.begin() + left_index, orig_y.begin() + left_index + num_datapoints, vec_y.begin());
2225 std::copy(orig_e.begin() + left_index, orig_e.begin() + left_index + num_datapoints, vec_e.begin());
2238 size_t left_index, right_index;
2242 if (!estimateBackgroundParameters(
m_inputMatrixWS->histogram(iws), std::pair<size_t, size_t>(left_index, right_index),
2247 std::vector<double> vec_x, vec_y, vec_e;
2253 reduceByBackground(bkgd_function, vec_x, vec_y);
2256 auto it_max = std::max_element(vec_y.begin(), vec_y.end());
2257 double signal = vec_y[it_max - vec_y.begin()];
2258 if (signal <= DBL_MIN)
2263 double noise = estimateBackgroundNoise(vec_y);
2264 if (noise <= DBL_MIN)
2268 return signal / noise;
2276 std::stringstream errss;
2277 errss <<
"Workspace index " << wi <<
" is out of range "
2279 throw std::runtime_error(errss.str());
2286 std::stringstream errss;
2287 errss <<
"Peak index " << ipeak <<
" is out of range (" <<
m_numPeaksToFit <<
")";
2288 throw std::runtime_error(errss.str());
2294 std::stringstream errss;
2295 errss <<
"Peak window is inappropriate for workspace index: " <<
left <<
" >= " <<
right;
2296 throw std::runtime_error(errss.str());
2310 const std::shared_ptr<FitPeaksAlgorithm::PeakFitResult> &fit_result) {
2314 g_log.
error() <<
"workspace index " << wi <<
" is out of output peak position workspace "
2318 throw std::runtime_error(
"Out of boundary to set output peak position workspace");
2323 double exp_peak_pos(expected_positions[ipeak]);
2324 double fitted_peak_pos = fit_result->getPeakPosition(ipeak);
2325 double peak_chi2 = fit_result->getCost(ipeak);
2344 <<
" parameters. Parameter table shall have 3 more "
2345 "columns. But not it has "
2347 throw std::runtime_error(
"Peak parameter vector for one peak has different sizes to output "
2354 std::stringstream err_ss;
2355 err_ss <<
"Peak has 4 effective peak parameters and " <<
m_bkgdFunction->nParams() <<
" background parameters "
2356 <<
". Parameter table shall have 3 more columns. But not it has " <<
m_fittedParamTable->columnCount()
2358 throw std::runtime_error(err_ss.str());
2365 size_t num_peakfunc_params = peak_function->nParams();
2375 for (
size_t iparam = 0; iparam < num_peakfunc_params + num_bkgd_params; ++iparam) {
2376 size_t col_index = iparam + 2;
2378 m_fittedParamTable->cell<
double>(row_index, col_index) = fit_result->getParameterValue(ipeak, iparam);
2381 m_fitErrorTable->cell<
double>(row_index, col_index) = fit_result->getParameterError(ipeak, iparam);
2388 for (
size_t iparam = 0; iparam < num_peakfunc_params; ++iparam)
2389 peak_function->setParameter(iparam, fit_result->getParameterValue(ipeak, iparam));
2392 peak_function->setMatrixWorkspace(
m_inputMatrixWS, wi, peak_window.first, peak_window.second);
2401 for (
size_t iparam = 0; iparam < num_bkgd_params; ++iparam)
2403 fit_result->getParameterValue(ipeak, num_peakfunc_params + iparam);
2407 m_fittedParamTable->cell<
double>(row_index, chi2_index) = fit_result->getCost(ipeak);
2415 std::string height_name(
"");
2417 std::vector<std::string> peak_parameters = peak_function->getParameterNames();
2418 for (
const auto &parName : peak_parameters) {
2419 if (parName ==
"Height") {
2420 height_name =
"Height";
2422 }
else if (parName ==
"I") {
2425 }
else if (parName ==
"Intensity") {
2426 height_name =
"Intensity";
2431 if (height_name.empty())
2432 throw std::runtime_error(
"Peak height parameter name cannot be found.");
2440 LoggingOffsetSentry sentry(
this);
#define DECLARE_ALGORITHM(classname)
double value
The value of the point.
#define PARALLEL_START_INTERRUPT_REGION
Begins a block to skip processing is the algorithm has been interupted Note the end of the block if n...
#define PARALLEL_CRITICAL(name)
#define PARALLEL_END_INTERRUPT_REGION
Ends a block to skip processing is the algorithm has been interupted Note the start of the block if n...
#define PARALLEL_FOR_IF(condition)
Empty definitions - to enable set your complier to enable openMP.
#define PRAGMA_OMP(expression)
#define PARALLEL_CHECK_INTERRUPT_REGION
Adds a check after a Parallel region to see if it was interupted.
Base class from which all concrete algorithm classes should be derived.
void declareProperty(std::unique_ptr< Kernel::Property > p, const std::string &doc="") override
Add a property to the list of managed properties.
std::string getPropertyValue(const std::string &name) const override
Get the value of a property as a string.
TypedValue getProperty(const std::string &name) const override
Get the value of a property.
virtual std::shared_ptr< Algorithm > createChildAlgorithm(const std::string &name, const double startProgress=-1., const double endProgress=-1., const bool enableLogging=true, const int &version=-1)
Create a Child Algorithm.
void setLoggingOffset(const int value) override
gets the logging priority offset
int getLoggingOffset() const override
returns the logging priority offset
bool isDefault(const std::string &name) const
static bool isEmpty(const NumT toCheck)
checks that the value was not set by users, uses the value in empty double/int.
Implements FunctionDomain1D with its own storage in form of a std::vector.
A class to store values calculated by a function.
const std::vector< double > & toVector() const
Return the calculated values as a vector.
double getCalculated(size_t i) const
Get i-th calculated value.
An interface to a peak function, which extend the interface of IFunctionWithLocation by adding method...
Helper class for reporting progress from algorithms.
TableRow represents a row in a TableWorkspace.
A property class for workspaces.
size_t m_submitted_spectrum_peaks
void setNumberOfSpectrumPeaksWithLowCount(const size_t n)
std::string getReport() const
size_t m_low_count_individual
PeakFitPreCheckResult & operator+=(const PeakFitPreCheckResult &another)
size_t m_not_enough_datapoints
void setNumberOfSubmittedIndividualPeaks(const size_t n)
void setNumberOfPeaksWithNotEnoughDataPoints(const size_t n)
size_t m_low_count_spectrum
void setNumberOfOutOfRangePeaks(const size_t n)
void setNumberOfPeaksWithLowSignalToNoise(const size_t n)
size_t m_submitted_individual_peaks
void setNumberOfIndividualPeaksWithLowCount(const size_t n)
void setNumberOfSubmittedSpectrumPeaks(const size_t n)
bool isIndividualPeakRejected() const
size_t m_function_parameters_number
number of function parameters
double getPeakPosition(size_t ipeak) const
size_t getNumberPeaks() const
std::vector< std::vector< double > > m_function_parameters_vector
PeakFitResult(size_t num_peaks, size_t num_params)
Holds all of the fitting information for a single spectrum.
double getParameterValue(size_t ipeak, size_t iparam) const
get the fitted value of a particular parameter
std::vector< double > m_costs
std::vector< std::vector< double > > m_function_errors_vector
fitted peak and background parameters' fitting error
size_t getNumberParameters() const
double getCost(size_t ipeak) const
void setRecord(size_t ipeak, const double cost, const double peak_position, const FitFunction &fit_functions)
set the peak fitting record/parameter for one peak
double getParameterError(size_t ipeak, size_t iparam) const
get the fitting error of a particular parameter
void setBadRecord(size_t ipeak, const double peak_position)
The peak postition should be negative and indicates what went wrong.
std::vector< double > m_fitted_peak_positions
Algorithms::PeakParameterHelper::EstimatePeakWidth m_peakWidthEstimateApproach
Flag for observing peak width: there are 3 states (1) no estimation (2) from 'observation' (3) calcul...
void calculateFittedPeaks(const std::vector< std::shared_ptr< FitPeaksAlgorithm::PeakFitResult > > &fit_results)
calculate peak+background for fitted
double calculateSignalToNoiseRatio(size_t iws, const std::pair< double, double > &range, const API::IBackgroundFunction_sptr &bkgd_function)
calculate signal-to-noise ratio in histogram range
API::MatrixWorkspace_const_sptr m_peakCenterWorkspace
void generateFittedParametersValueWorkspaces()
Generate output workspaces.
void setupParameterTableWorkspace(const API::ITableWorkspace_sptr &table_ws, const std::vector< std::string > ¶m_names, bool with_chi2)
Set up parameter table (parameter value or error)
bool m_copyLastGoodPeakParameters
std::vector< double > m_peakPosTolerances
tolerances for fitting peak positions
API::IPeakFunction_sptr m_peakFunction
Peak profile name.
bool fitBackground(const size_t &ws_index, const std::pair< double, double > &fit_window, const double &expected_peak_pos, const API::IBackgroundFunction_sptr &bkgd_func)
fit background
double fitFunctionSD(const API::IAlgorithm_sptr &fit, const API::IPeakFunction_sptr &peak_function, const API::IBackgroundFunction_sptr &bkgd_function, const API::MatrixWorkspace_sptr &dataws, size_t wsindex, const std::pair< double, double > &peak_range, const double &expected_peak_center, bool estimate_peak_width, bool estimate_background)
Methods to fit functions (general)
API::MatrixWorkspace_sptr m_outputPeakPositionWorkspace
output workspace for peak positions
API::MatrixWorkspace_sptr m_fittedPeakWS
matrix workspace contained calcalated peaks+background from fitted result it has same number of spect...
API::ITableWorkspace_const_sptr m_profileStartingValueTable
table workspace for profile parameters' starting value
std::string m_minimizer
Minimzer.
bool m_respectFixedPeakParameters
API::IBackgroundFunction_sptr m_linearBackgroundFunction
Linear background function for high background fitting.
bool m_constrainPeaksPosition
bool m_peakPosTolCase234
peak positon tolerance case b, c and d
void fitSpectrumPeaks(size_t wi, const std::vector< double > &expected_peak_centers, const std::shared_ptr< FitPeaksAlgorithm::PeakFitResult > &fit_result, std::vector< std::vector< double > > &lastGoodPeakParameters, std::vector< size_t > &lastGoodPeakSpectra, const std::shared_ptr< FitPeaksAlgorithm::PeakFitPreCheckResult > &pre_check_result)
fit peaks in a same spectrum
std::map< std::string, std::string > validateInputs() override
Validate inputs.
double m_peakWidthPercentage
flag to estimate peak width from
API::IBackgroundFunction_sptr m_bkgdFunction
Background function.
size_t histRangeToDataPointCount(size_t iws, const std::pair< double, double > &range)
convert a histogram range to index boundaries
void checkPeakIndices(std::size_t const &, std::size_t const &)
double fitFunctionHighBackground(const API::IAlgorithm_sptr &fit, const std::pair< double, double > &fit_window, const size_t &ws_index, const double &expected_peak_center, bool observe_peak_shape, const API::IPeakFunction_sptr &peakfunction, const API::IBackgroundFunction_sptr &bkgdfunc)
fit a single peak with high background
void processInputPeakCenters()
peak centers
std::vector< std::shared_ptr< FitPeaksAlgorithm::PeakFitResult > > fitPeaks()
suites of method to fit peaks
double m_minPeakHeight
minimum peak height without background and it also serves as the criteria for observed peak parameter
std::string m_costFunction
Cost function.
void processInputFunctions()
process inputs for peak and background functions
void processInputPeakTolerance()
process inputs about fitted peak positions' tolerance
void writeFitResult(size_t wi, const std::vector< double > &expected_positions, const std::shared_ptr< FitPeaksAlgorithm::PeakFitResult > &fit_result)
Write result of peak fit per spectrum to output analysis workspaces.
void histRangeToIndexBounds(size_t iws, const std::pair< double, double > &range, size_t &left_index, size_t &right_index)
convert a histogram range to index boundaries
void exec() override
Main exec method.
void init() override
Init.
void logNoOffset(const size_t &priority, const std::string &msg)
std::size_t m_numSpectraToFit
total number of spectra to be fit
std::vector< std::string > m_peakParamNames
input peak parameters' names
double calculateSignalToSigmaRatio(const size_t &iws, const std::pair< double, double > &peakWindow, const API::IPeakFunction_sptr &peakFunction)
bool isObservablePeakProfile(const std::string &peakprofile)
check whether FitPeaks supports observation on a certain peak profile's parameters (width!...
void processInputFitRanges()
process inputs for peak fitting range
std::size_t m_numPeaksToFit
the number of peaks to fit in all spectra
std::size_t m_startWorkspaceIndex
start index
API::MatrixWorkspace_sptr createMatrixWorkspace(const std::vector< double > &vec_x, const std::vector< double > &vec_y, const std::vector< double > &vec_e)
Create a single spectrum workspace for fitting.
void generateOutputPeakPositionWS()
main method to create output workspaces
void generateCalculatedPeaksWS()
Generate workspace for calculated values.
double fitFunctionMD(API::IFunction_sptr fit_function, const API::MatrixWorkspace_sptr &dataws, const size_t wsindex, const std::pair< double, double > &vec_xmin, const std::pair< double, double > &vec_xmax)
double m_minPeakTotalCount
std::string getPeakHeightParameterName(const API::IPeakFunction_const_sptr &peak_function)
Get the parameter name for peak height (I or height or etc)
bool m_uniformPeakPositions
API::ITableWorkspace_sptr m_fittedParamTable
output analysis workspaces table workspace for fitted parameters
double m_minSignalToNoiseRatio
bool decideToEstimatePeakParams(const bool firstPeakInSpectrum, const API::IPeakFunction_sptr &peak_function)
Decide whether to estimate peak parameters.
void convertParametersNameToIndex()
Convert peak function's parameter names to parameter index for fast access.
void processOutputs(std::vector< std::shared_ptr< FitPeaksAlgorithm::PeakFitResult > > fit_result_vec)
Set the workspaces and etc to output properties.
bool m_highBackground
flag for high background
std::vector< double > m_initParamValues
input peak parameters' starting values corresponding to above peak parameter names
std::vector< double > m_peakCenters
Designed peak positions and tolerance.
double fitIndividualPeak(size_t wi, const API::IAlgorithm_sptr &fitter, const double expected_peak_center, const std::pair< double, double > &fitwindow, const bool estimate_peak_width, const API::IPeakFunction_sptr &peakfunction, const API::IBackgroundFunction_sptr &bkgdfunc, const std::shared_ptr< FitPeaksAlgorithm::PeakFitPreCheckResult > &pre_check_result)
Fit an individual peak.
std::vector< std::vector< double > > m_peakWindowVector
peak windows
std::function< std::pair< double, double >(std::size_t const &, std::size_t const &)> m_getPeakFitWindow
API::MatrixWorkspace_sptr m_inputMatrixWS
mandatory input and output workspaces
std::vector< size_t > m_initParamIndexes
input starting parameters' indexes in peak function
bool m_fitPeaksFromRight
Fit from right or left.
std::function< std::vector< double >(std::size_t const &)> m_getExpectedPeakPositions
bool m_rawPeaksTable
flag to show that the pamarameters in table are raw parameters or effective parameters
void processInputs()
process inputs (main and child algorithms)
void checkWorkspaceIndices(std::size_t const &)
Get the expected peak's position.
void checkPeakWindowEdgeOrder(double const &, double const &)
int m_fitIterations
Fit iterations.
double m_minSignalToSigmaRatio
double numberCounts(size_t iws)
sum up all counts in histogram
API::MatrixWorkspace_const_sptr m_peakWindowWorkspace
bool m_uniformProfileStartingValue
flag for profile startng value being uniform or not
API::ITableWorkspace_sptr m_fitErrorTable
table workspace for fitted parameters' fitting error. This is optional
void getRangeData(size_t iws, const std::pair< double, double > &range, std::vector< double > &vec_x, std::vector< double > &vec_y, std::vector< double > &vec_e)
get vector X, Y and E in a given range
std::size_t m_stopWorkspaceIndex
stop index (workspace index of the last spectrum included)
bool processSinglePeakFitResult(size_t wsindex, size_t peakindex, const double cost, const std::vector< double > &expected_peak_positions, const FitPeaksAlgorithm::FitFunction &fitfunction, const std::shared_ptr< FitPeaksAlgorithm::PeakFitResult > &fit_result)
Process the result from fitting a single peak.
Support for a property that holds an array of values.
Exception for when an item is not found in a collection.
IPropertyManager * setProperty(const std::string &name, const T &value)
Templated method to set the value of a PropertyWithValue.
void setPropertyGroup(const std::string &name, const std::string &group)
Set the group for a given property.
ListValidator is a validator that requires the value of a property to be one of a defined list of pos...
void debug(const std::string &msg)
Logs at debug level.
void notice(const std::string &msg)
Logs at notice level.
void error(const std::string &msg)
Logs at error level.
void warning(const std::string &msg)
Logs at warning level.
void report()
Increments the loop counter by 1, then sends the progress notification on behalf of its algorithm.
StartsWithValidator is a validator that requires the value of a property to start with one of the str...
std::shared_ptr< IAlgorithm > IAlgorithm_sptr
shared pointer to Mantid::API::IAlgorithm
std::shared_ptr< IBackgroundFunction > IBackgroundFunction_sptr
std::shared_ptr< IPeakFunction > IPeakFunction_sptr
std::shared_ptr< ITableWorkspace > ITableWorkspace_sptr
shared pointer to Mantid::API::ITableWorkspace
std::shared_ptr< const IPeakFunction > IPeakFunction_const_sptr
std::shared_ptr< const MatrixWorkspace > MatrixWorkspace_const_sptr
shared pointer to the matrix workspace base class (const version)
std::shared_ptr< IFunction > IFunction_sptr
shared pointer to the function base class
std::shared_ptr< MatrixWorkspace > MatrixWorkspace_sptr
shared pointer to the matrix workspace base class
std::shared_ptr< CompositeFunction > CompositeFunction_sptr
shared pointer to the composite function base class
std::string const OUTPUT_WKSP("OutputWorkspace")
std::string const INPUT_WKSP("InputWorkspace")
MANTID_ALGORITHMS_DLL size_t findXIndex(const vector_like &vecx, const double x, const size_t startindex=0)
Get an index of a value in a sorted vector.
MANTID_ALGORITHMS_DLL int estimatePeakParameters(const HistogramData::Histogram &histogram, const std::pair< size_t, size_t > &peak_window, const API::IPeakFunction_sptr &peakfunction, const API::IBackgroundFunction_sptr &bkgdfunction, bool observe_peak_width, const EstimatePeakWidth peakWidthEstimateApproach, const double peakWidthPercentage, const double minPeakHeight)
Estimate peak parameters by 'observation'.
Statistics getStatistics(const std::vector< TYPE > &data, const unsigned int flags=StatOptions::AllStats)
Return a statistics object for the given data set.
std::vector< double > getZscore(const std::vector< TYPE > &data)
Return the Z score values for a dataset.
std::shared_ptr< IValidator > IValidator_sptr
A shared_ptr to an IValidator.
std::enable_if< std::is_pointer< Arg >::value, bool >::type threadSafe(Arg workspace)
Thread-safety check Checks the workspace to ensure it is suitable for multithreaded access.
const std::string OUTPUT_WKSP("OutputWorkspace")
const std::string INPUT_WKSP("InputWorkspace")
Helper class which provides the Collimation Length for SANS instruments.
constexpr int EMPTY_INT() noexcept
Returns what we consider an "empty" integer within a property.
constexpr double EMPTY_DBL() noexcept
Returns what we consider an "empty" double within a property.
API::IPeakFunction_sptr peakfunction
API::IBackgroundFunction_sptr bkgdfunction
@ Input
An input workspace.
@ Output
An output workspace.
Functor to accumulate a sum of squares.