Skip to content
Snippets Groups Projects
MuonFitPropertyBrowser.cpp 64.3 KiB
Newer Older

  // Group output together
  std::string groupName = fitAlg->getPropertyValue("Output");
  const std::string &baseName = groupName;

  // Create a group for label
  try {
    ads.addOrReplace(groupName, boost::make_shared<WorkspaceGroup>());
    ads.addToGroup(groupName, baseName + "_NormalisedCovarianceMatrix");
    ads.addToGroup(groupName, baseName + "_Parameters");
    ads.addToGroup(groupName, baseName + "_Workspaces");
  } catch (const Mantid::Kernel::Exception::NotFoundError &err) {
    g_log.warning(err.what());
  }
 * After a TF simultaneous fit, insert extra information into parameters table
 * (i.e. what runs, groups, periods "f0", "f1" etc were)
 * and group the output workspaces
 * @param alg :: [input] Pointer to fit algorithm that just finished
 * @param baseName :: [input] The common name of the workspaces of interest
 */
void MuonFitPropertyBrowser::finishAfterTFSimultaneousFit(
    const Mantid::API::IAlgorithm *alg, const std::string baseName) const {
  AnalysisDataServiceImpl &ads = AnalysisDataService::Instance();
  try {
    std::vector<std::string> wsList =
        alg->getProperty("UnNormalizedWorkspaceList");
    std::string paramTableName = baseName + "_Parameters";
    const auto paramTable = ads.retrieveWS<ITableWorkspace>(paramTableName);
    if (paramTable) {
      for (size_t i = 0; i < wsList.size(); i++) {
        const std::string suffix = boost::lexical_cast<std::string>(i);

        const auto wsName = wsList[i];
        Mantid::API::TableRow row = paramTable->appendRow();
        row << "f" + suffix + "=" + wsName << 0.0 << 0.0;
      }
    }
  } catch (const Mantid::Kernel::Exception::NotFoundError &) {
    // Not a fatal error, but shouldn't happen
    g_log.warning(
        "Could not find output parameters table for simultaneous fit");
  }

  // Group output together

  std::string groupName = baseName;
  // Create a group for label
  try {
    ads.addOrReplace(groupName, boost::make_shared<WorkspaceGroup>());
    ads.addToGroup(groupName, baseName + "_NormalisedCovarianceMatrix");
    ads.addToGroup(groupName, baseName + "_Parameters");
    ads.addToGroup(groupName, baseName + "_Workspaces");
  } catch (const Mantid::Kernel::Exception::NotFoundError &err) {
    g_log.warning(err.what());
  }
/**
 * Adds an extra widget in between the fit buttons and the browser
 * @param widget :: [input] Pointer to widget to add
 */
void MuonFitPropertyBrowser::addExtraWidget(QWidget *widget) {
  widget->setSizePolicy(QSizePolicy::Policy::Expanding,
                        QSizePolicy::Policy::Expanding);
Tom Perkins's avatar
Tom Perkins committed
  if (m_widgetSplitter) {
    m_widgetSplitter->addWidget(widget);
 * Called externally to set the function
 * @param func :: [input] Fit function to use
void MuonFitPropertyBrowser::setFunction(const IFunction_sptr func) {
  createCompositeFunction(func);
/**
 * Set the list of workspaces to fit to the given list
 * @param wsNames :: [input] List of workspace names to fit
 */
void MuonFitPropertyBrowser::setWorkspaceNames(const QStringList &wsNames) {
  m_workspacesToFit.clear();
  std::transform(wsNames.begin(), wsNames.end(),
                 std::back_inserter(m_workspacesToFit),
                 [](const QString &qs) { return qs.toStdString(); });
  // Update listeners
  emit workspacesToFitChanged(static_cast<int>(m_workspacesToFit.size()));
  // Update norm
  setNormalization();
/**
 * Override in the case of simultaneous fits to use a special prefix.
 * Otherwise, use the parent class method.
 * @returns :: output name for Fit algorithm
 */
std::string MuonFitPropertyBrowser::outputName() const {
  const int nWorkspaces = static_cast<int>(m_workspacesToFit.size());
  if (nWorkspaces > 1) {
    // simultaneous fit
    return SIMULTANEOUS_PREFIX + m_simultaneousLabel;
  } else {
    // use parent class behaviour
    return FitPropertyBrowser::outputName();
  }
}

 * Set multiple fitting mode on or off.
 * If turned off, all parts of the fit property browser are shown and all extra
 * widgets (like the function browser or data selector) are hidden, so it looks
 * just like it used to before the changes in Mantid 3.8.
 * If turned on, the "Function" and "Data" sections of the fit property browser
 * are hidden and the extra widgets are shown.
 * @param enabled :: [input] Whether to turn this mode on or off
 */
void MuonFitPropertyBrowser::setMultiFittingMode(bool enabled) {
  m_isMultiFittingMode = enabled;
  // First, clear whatever model is currently set
  this->clear();
  // set default selection (all groups)
  if (enabled) {
Anthony Lim's avatar
Anthony Lim committed
    setAllGroups();
Anthony Lim's avatar
Anthony Lim committed
    setAllPeriods();
    this->clear(); // force update of composite function
  } else {         // clear current selection
    if (m_autoBackground != "") {
      setAutoBackgroundName(m_autoBackground);
      addAutoBackground();
    }
Anthony Lim's avatar
Anthony Lim committed
    clearChosenGroups();
    clearChosenPeriods();
  // Show or hide "Function" and "Data" sections
  m_browser->setItemVisible(m_functionsGroup, !enabled);
  m_browser->setItemVisible(m_settingsGroup, !enabled);
  m_browser->setItemVisible(m_multiFitSettingsGroup, enabled);
  m_btnGroup->setVisible(enabled);
  // Show or hide additional widgets
Tom Perkins's avatar
Tom Perkins committed
  for (int i = 0; i < m_widgetSplitter->count(); ++i) {
    if (auto *widget = m_widgetSplitter->widget(i)) {
      widget->setVisible(enabled);
  if (enabled) {
    setFitEnabled(false);
  }
 * Returns true is the browser is set to multi fitting mode
 * This works using the visibility state of the button group
 * which is controlled in setMultiFittingMode
 */
bool MuonFitPropertyBrowser::isMultiFittingMode() const {
  return m_isMultiFittingMode;
}
void MuonFitPropertyBrowser::ConvertFitFunctionForMuonTFAsymmetry(
    bool enabled) {
  // set new fit func
  IAlgorithm_sptr alg = AlgorithmManager::Instance().create(
      "ConvertFitFunctionForMuonTFAsymmetry");
  // do not preserve the ties
  if (AnalysisDataService::Instance().doesExist(
          "MuonAnalysisTFNormalizations") &&
      m_compositeFunction->nFunctions() > 0) {
    alg->initialize();
    IFunction_sptr old =
        boost::dynamic_pointer_cast<IFunction>(m_compositeFunction);
    QStringList globals;

    if ( m_isMultiFittingMode) {
      // manually set the function values
      old = m_functionBrowser->getGlobalFunction();
      globals = m_functionBrowser->getGlobalParameters();
    } else if (!enabled && !m_isMultiFittingMode) {
      // to extract in single fit we have an extra composite -> so remove it
      auto tmp = boost::dynamic_pointer_cast<CompositeFunction>(old);
      old = tmp->getFunction(0);
    }
    alg->setProperty("InputFunction", old);
    alg->setProperty("NormalizationTable", "MuonAnalysisTFNormalizations");
    alg->setProperty("WorkspaceList", m_workspacesToFit);
    std::string mode = (enabled) ? "Construct" : "Extract";
    alg->setProperty("Mode", mode);
    alg->execute();
    if (!alg->isExecuted()) {
      return;
    }
    IFunction_sptr func = alg->getProperty("OutputFunction");

    // multiple fit
    if (m_isMultiFittingMode) {
      // update values in browser
      if (func->getNumberDomains() > 1) {
        auto tmp = boost::dynamic_pointer_cast<MultiDomainFunction>(func);
        old = tmp->getFunction(0);
      } else {
        old = func;
      }
      m_functionBrowser->setFunction(old);
      // preserve global parameters
      QStringList newGlobals;
      const std::string INSERT_FUNCTION{"f0.f1.f1."};
      if (enabled) {
        for (auto global : globals) {
          newGlobals << QString::fromStdString(INSERT_FUNCTION) + global;
        }
      } else {
        for (auto global : globals) {
          newGlobals << global.remove(0, 9);
        }
      }
      m_functionBrowser->updateMultiDatasetParameters(*func);

      m_functionBrowser->setGlobalParameters(newGlobals);
      // if multi data set we need to do the fixes manually
      // the current domain is automatic
      auto originalNames = func->getParameterNames();
      for (auto name : originalNames) {
        auto index = func->parameterIndex(name);
        if (func->isFixed(index) && func->getNumberDomains() > 1) {
          // get domain
          auto index = name.find_first_of(".");
          std::string domainStr = name.substr(1, index - 1);
          int domain = std::stoi(domainStr);
          // remove domain from name
          auto newName = name.substr(index + 1);
          // set fix
          m_functionBrowser->setLocalParameterFixed(
              QString::fromStdString(newName), domain, true);
      }
    } // single fit
    else {
      FitPropertyBrowser::clear();
      FitPropertyBrowser::addFunction(func->asString());
    }

    updateTFPlot();
    // m_enumManager->setValue(m_workspace,j);
  }
}
 * Set TF asymmetry mode on or off.
 * If turned off, the fit property browser looks like Mantid 3.8.
 * If turned on, the fit menu has an extra button and
 * normalization is shown in the data table
 * @param enabled :: [input] Whether to turn this mode on or off
 */
void MuonFitPropertyBrowser::setTFAsymmMode(bool enabled) {
  IFunction_sptr old =
      boost::dynamic_pointer_cast<IFunction>(m_compositeFunction);
  if (old->nParams() > 0) {
    ConvertFitFunctionForMuonTFAsymmetry(enabled);
    // Show or hide the TFAsymmetry fit
    if (enabled) {
      m_settingsGroup->property()->addSubProperty(m_keepNorm);
    } else {
      m_settingsGroup->property()->removeSubProperty(m_keepNorm);
    }
  } else if (enabled) {
    // will update when user clicks elsewhere
    m_boolManager->setValue(m_TFAsymmMode, false);
    QMessageBox::warning(this, "Muon Analysis",
                         "No fitting function provided. TF Asymmetry mode "
                         "requires a fitting function to be added before "
                         "enabling. Please add a fitting function and enable "
                         "TF Asymmetry Mode again.");
std::string MuonFitPropertyBrowser::TFExtension() const {

  return (m_boolManager->value(m_TFAsymmMode)) ? UNNORM : "";
 * Makes sure we have the TF plot in TFAsymm mode
 */
void MuonFitPropertyBrowser::updateTFPlot() {
  // update plot
  int j = m_enumManager->value(m_workspace);
  std::string option = m_workspaceNames[j].toStdString();
  if (m_boolManager->value(m_TFAsymmMode) &&
      option.find(UNNORM) == std::string::npos) {
    auto raw = option.find("_Raw");

    if (raw == std::string::npos) {
      option += TFExtension();
    } else {
      option.insert(raw, UNNORM);
    }
  }
  // update plot
  emit TFPlot(QString::fromStdString(option));
/**
 * Adds an extra widget in between the fit buttons and the browser
 * @param widget :: [input] Pointer to widget to add
 * @param functionBrowser :: [input] pointer to the function browser
void MuonFitPropertyBrowser::addFitBrowserWidget(
    QWidget *widget,
    MantidQt::MantidWidgets::FunctionBrowser *functionBrowser) {
  widget->setSizePolicy(QSizePolicy::Policy::Expanding,
                        QSizePolicy::Policy::Expanding);
  if (m_widgetSplitter) {
    m_widgetSplitter->addWidget(widget);
  }
  m_functionBrowser = functionBrowser;
/**
 * The pre-fit checks have been successfully completed. Continue by emitting a
 * signal to update the function and request the fit.
 * @param sequential :: [input] Whether fit is sequential or not
 */
void MuonFitPropertyBrowser::continueAfterChecks(bool sequential) {
Raquel Alvarez's avatar
Raquel Alvarez committed
  emit functionUpdateAndFitRequested(sequential);
/**
 * Returns whether or not a guess is plotted
 * @returns :: True if a plot guess is plotted, false if not.
 */
bool MuonFitPropertyBrowser::hasGuess() const {
  auto *handler = getHandler();
  if (handler) {
    const bool hasPlot = handler->hasPlot(); // don't allow caller to modify
    return hasPlot;
  } else {
    return false;
  }
 * Sets group names and updates checkboxes on UI
 * By default sets all unchecked
 * @param groups :: [input] List of group names
 */
void MuonFitPropertyBrowser::setAvailableGroups(const QStringList &groups) {
Anthony Lim's avatar
Anthony Lim committed
  // If it's the same list, do nothing
  auto selected = getChosenGroups();
Anthony Lim's avatar
Anthony Lim committed
  if (groups.size() == m_groupBoxes.size()) {
    auto existingGroups = m_groupBoxes.keys();
    auto newGroups = groups;
    qSort(existingGroups);
    qSort(newGroups);
    if (existingGroups == newGroups) {
      return;
    }
  }
  clearGroupCheckboxes();
  QSettings settings;
  for (const auto &group : groups) {
Anthony Lim's avatar
Anthony Lim committed
    addGroupCheckbox(group);
  }
  // sets the same selection as before
  for (const auto &group : selected) {

    for (auto iter = m_groupBoxes.constBegin(); iter != m_groupBoxes.constEnd();
         ++iter) {
      if (iter.key().toStdString() == group.toStdString()) {
        m_boolManager->setValue(iter.value(), true);
      }
    }
 * Selects a single group/pair
 * @param group :: [input] Group/pair to select
 */
void MuonFitPropertyBrowser::setChosenGroup(const QString &group) {
Anthony Lim's avatar
Anthony Lim committed
  clearChosenGroups();
  for (auto iter = m_groupBoxes.constBegin(); iter != m_groupBoxes.constEnd();
       ++iter) {
    if (iter.key() == group) {
      m_boolManager->setValue(iter.value(), true);
    }
  }
 * Clears all group names and checkboxes
 * (ready to add new ones)
 */
void MuonFitPropertyBrowser::clearGroupCheckboxes() {
Anthony Lim's avatar
Anthony Lim committed
  for (const auto &checkbox : m_groupBoxes) {
    delete (checkbox);
  }
  m_groupBoxes.clear();
 * Add a new checkbox to the list of groups with given name
 * The new checkbox is checked according to dropdown menu selection
 * @param name :: [input] Name of group to add
 */
void MuonFitPropertyBrowser::addGroupCheckbox(const QString &name) {
Anthony Lim's avatar
Anthony Lim committed
  m_groupBoxes.insert(name, m_boolManager->addProperty(name));
  int j = m_enumManager->value(m_groupsToFit);
  auto option = m_groupsToFitOptions[j].toStdString();
 * Returns a list of the selected groups (checked boxes)
 * @returns :: list of selected groups
 */
QStringList MuonFitPropertyBrowser::getChosenGroups() const {
Anthony Lim's avatar
Anthony Lim committed
  QStringList chosen;
  for (auto iter = m_groupBoxes.constBegin(); iter != m_groupBoxes.constEnd();
       ++iter) {
    if (m_boolManager->value(iter.value()) == true) {
      chosen.append(iter.key());
    }
  }
  return chosen;
 * Clears the list of selected groups (unchecks boxes)
 */
void MuonFitPropertyBrowser::clearChosenGroups() const {
Anthony Lim's avatar
Anthony Lim committed
  for (auto iter = m_groupBoxes.constBegin(); iter != m_groupBoxes.constEnd();
       ++iter) {
    m_boolManager->setValue(iter.value(), false);
  }
 * Selects all groups
 */
void MuonFitPropertyBrowser::setAllGroups() {
Anthony Lim's avatar
Anthony Lim committed

  clearChosenGroups();
  for (auto iter = m_groupBoxes.constBegin(); iter != m_groupBoxes.constEnd();
       ++iter) {
    for (const auto &group : m_groupsList) {
Anthony Lim's avatar
Anthony Lim committed
      if (iter.key().toStdString() == group) {
        m_boolManager->setValue(iter.value(), true);
      }
    }
  }
 * Sets all pairs
 */
void MuonFitPropertyBrowser::setAllPairs() {
Anthony Lim's avatar
Anthony Lim committed
  clearChosenGroups();
  for (auto iter = m_groupBoxes.constBegin(); iter != m_groupBoxes.constEnd();
       ++iter) {
    bool isItGroup = false;
    for (const auto &group : m_groupsList) {
Anthony Lim's avatar
Anthony Lim committed
      if (iter.key().toStdString() == group) {
        isItGroup = true;
      }
    }
    if (!isItGroup) {
      m_boolManager->setValue(iter.value(), true);
    }
  }
 * Create a popup window to select a custom
 * selection of groups/pairs
 */
void MuonFitPropertyBrowser::genGroupWindow() {
Anthony Lim's avatar
Anthony Lim committed
  // reset group window
  m_groupWindow = new QDialog(this);
Anthony Lim's avatar
Anthony Lim committed
  QtGroupPropertyManager *groupManager =
      new QtGroupPropertyManager(m_groupWindow);
  QVBoxLayout *layout = new QVBoxLayout(m_groupWindow);
  QtTreePropertyBrowser *groupBrowser = new QtTreePropertyBrowser();
  QtProperty *groupSettings = groupManager->addProperty("Group/Pair selection");
  for (auto iter = m_groupBoxes.constBegin(); iter != m_groupBoxes.constEnd();
       ++iter) {
    groupSettings->addSubProperty(m_groupBoxes.value(iter.key()));
    m_boolManager->setValue(iter.value(), m_boolManager->value(iter.value()));
  }
  QtCheckBoxFactory *checkBoxFactory = new QtCheckBoxFactory(m_groupWindow);
  groupBrowser->setFactoryForManager(m_boolManager, checkBoxFactory);
  groupBrowser->addProperty(groupSettings);
  layout->addWidget(groupBrowser);
  m_groupWindow->setLayout(layout);
  m_groupWindow->show();
 * Selects all periods
 */
void MuonFitPropertyBrowser::setAllPeriods() {

Anthony Lim's avatar
Anthony Lim committed
  for (auto iter = m_periodBoxes.constBegin(); iter != m_periodBoxes.constEnd();
       ++iter) {
    m_boolManager->setValue(iter.value(), true);
  }
 * Sets checkboxes for periods
 * @param numPeriods :: [input] Number of periods
 */
void MuonFitPropertyBrowser::setNumPeriods(size_t numPeriods) {
  // has to go here to get the original value
  int j = m_enumManager->value(m_periodsToFit);
  auto selected = getChosenPeriods();
  // delete period checkboxes
  clearPeriodCheckboxes();
  if (!m_periodsToFitOptions.empty()) {
    m_periodsToFitOptions.clear();
  if (numPeriods > 1) {
    m_periodsToFitOptions << ALL_PERIODS_LABEL;
    m_periodsToFitOptions << CUSTOM_LABEL;
Anthony Lim's avatar
Anthony Lim committed
  // create more boxes
  for (size_t i = 0; i != numPeriods; i++) {
    QString name = QString::number(i + 1);
    addPeriodCheckbox(name);
  }
Anthony Lim's avatar
Anthony Lim committed
  if (m_periodsToFitOptions.size() == 1) {
    m_generateBtn->setDisabled(true);
    m_multiFitSettingsGroup->property()->removeSubProperty(m_periodsToFit);
    m_multiFitSettingsGroup->property()->removeSubProperty(m_showPeriods);
    m_enumManager->setValue(m_periodsToFit, 0);
    clearChosenPeriods();
    m_boolManager->setValue(m_periodBoxes.constBegin().value(), true);
  } else {
    // for now always reset to all groups when data is changed
    // the commented out code can be used to keep the selection when changing
    // run - but has a bug
    // if (j >= m_periodsToFitOptions.size()) {
    // set all groups if the selection is no longer available (0 index)
    j = 0;
    //}
Anthony Lim's avatar
Anthony Lim committed
    m_multiFitSettingsGroup->property()->insertSubProperty(m_periodsToFit,
                                                           m_showGroup);
    m_multiFitSettingsGroup->property()->addSubProperty(m_showPeriods);
    m_generateBtn->setDisabled(false);

    updatePeriods(j); // , selected);
 * Sets period names and updates checkboxes on UI
 * By default sets all unchecked
 * @param periods :: [input] List of period names
 */
void MuonFitPropertyBrowser::setAvailablePeriods(const QStringList &periods) {
Anthony Lim's avatar
Anthony Lim committed
  // If it's the same list, do nothing
  if (periods.size() == m_periodBoxes.size()) {
    auto existingGroups = m_periodBoxes.keys();
    auto newGroups = periods;
    qSort(existingGroups);
    qSort(newGroups);
    if (existingGroups == newGroups) {
      return;
    }
  }

  clearPeriodCheckboxes();

  for (const auto &group : periods) {
Anthony Lim's avatar
Anthony Lim committed
    addPeriodCheckbox(group);
  }
 * Clears all pair names and checkboxes
 * (ready to add new ones)
 */
void MuonFitPropertyBrowser::clearPeriodCheckboxes() {
Anthony Lim's avatar
Anthony Lim committed
  if (m_periodBoxes.size() > 1) {
    for (auto iter = std::next(m_periodBoxes.constBegin());
Anthony Lim's avatar
Anthony Lim committed
         iter != m_periodBoxes.constEnd(); ++iter) {
    m_periodBoxes.clear();
Anthony Lim's avatar
Anthony Lim committed
  }
  m_periodsToFitOptions.clear();
  m_periodsToFitOptions << "1";
  m_enumManager->setEnumNames(m_periodsToFit, m_periodsToFitOptions);
 * Clears the list of selected groups (unchecks boxes)
 */
void MuonFitPropertyBrowser::clearChosenPeriods() const {
Anthony Lim's avatar
Anthony Lim committed
  for (auto iter = m_periodBoxes.constBegin(); iter != m_periodBoxes.constEnd();
       ++iter) {
    m_boolManager->setValue(iter.value(), false);
  }
 * updates the period displays
 */
void MuonFitPropertyBrowser::updatePeriods() {
  int j = m_enumManager->value(m_periodsToFit);
  // auto selected = getChosenPeriods();
 * updates the period displays and conserves the selection
 * if selection is niot available default to all periods
 * @param j :: [input] index of selection in combobox
 * selected is an input for changing runs and preserving selection (list of
 * selected periods)
 * currently has a bug
 */
void MuonFitPropertyBrowser::updatePeriods(const int j) {
  // this is for switching but has a bug at the moment
  // const QStringList &selected) {
  if (m_periodsToFitOptions.size() == 0) {
    QMessageBox::warning(this, "Muon Analysis",
Nick Draper's avatar
Nick Draper committed
                         "Data not found. Please turn on the data archive, "
                         "using the Manage Directories button.");
  m_enumManager->setEnumNames(m_periodsToFit, m_periodsToFitOptions);
  m_enumManager->setValue(m_periodsToFit, j);
  if (m_periodsToFitOptions[j] == CUSTOM_LABEL) {
    // currently the below does not work reliably (if period arithmatic is
    // presemnt it gives bad results
    /*
    setChosenPeriods(selected);*/
    // lets default to all periods for now
    // explictly set all periods
    setAllPeriods();
  } else if (m_periodsToFitOptions[j] == ALL_PERIODS_LABEL) {
    // explictly set all periods
    setAllPeriods();
  } else { // single number
    setChosenPeriods(m_periodsToFitOptions[j]);
  }
 * Adds a new checkbox to the list of periods with given name
 * It updates the display
 * @param name :: [input] Name of period to add
 */
void MuonFitPropertyBrowser::addPeriodCheckboxToMap(const QString &name) {
  if (m_periodBoxes.find(name) != m_periodBoxes.end()) {
    // if the box already exists
    return;
  }
  // has to go here to get the original value
  int j = m_enumManager->value(m_periodsToFit);
  // auto selected = getChosenPeriods();
  addPeriodCheckbox(name);
 * Check if a period is valid
 * @param name :: [input] Name of period to add
 */
bool MuonFitPropertyBrowser::isPeriodValid(const QString &name) {
  // check period is sensible
  // no frational periods
  if (name.contains(".")) {
    return false;
  }
  // wshould only ever have 1 minus sign
  else if (name.count("-") > 1) {
    return false;
  } else {
    std::vector<std::string> numbers;
    std::string nameString = name.toStdString();
    boost::algorithm::split(numbers, nameString, boost::is_any_of(","));
    // loop over results
    for (auto value : numbers) {
      auto tmp = value.find("-");
      if (tmp != std::string::npos) {
        // find a minus sign
        auto before = value.substr(0, tmp);
        auto after = value.substr(tmp + 1);

      } else {
        try {
          boost::lexical_cast<int>(value);
          if (m_periodBoxes.find(QString::fromStdString(value)) ==
                  m_periodBoxes.end() &&
              numbers.size() > 1) {
            // if the box does not exist and there is more than 1 period in name
            return false;
          }
        } catch (boost::bad_lexical_cast) {
          // none int value
          return false;
        }
      }
    }
  }
  return true;
 * Add a new checkbox to the list of periods with given name
 * The new checkbox is unchecked by default
 * @param name :: [input] Name of period to add
 */
void MuonFitPropertyBrowser::addPeriodCheckbox(const QString &name) {
  // check period is sensible
  // no frational periods
  if (isPeriodValid(name)) {
    m_periodBoxes.insert(name, m_boolManager->addProperty(name));
    int j = m_enumManager->value(m_periodsToFit);
    // add new period to list will go after inital list
    m_periodsToFitOptions << name;
    auto active = getChosenPeriods();
    m_enumManager->setEnumNames(m_periodsToFit, m_periodsToFitOptions);
    setChosenPeriods(active);
    m_enumManager->setValue(m_periodsToFit, j);
 * Returns a list of the selected periods (checked boxes)
 * @returns :: list of selected periods
 */
QStringList MuonFitPropertyBrowser::getChosenPeriods() const {
Anthony Lim's avatar
Anthony Lim committed
  QStringList chosen;
  // if single period
  if (m_periodsToFitOptions.size() == 1) {
    chosen << "";
  } else {
    for (auto iter = m_periodBoxes.constBegin();
         iter != m_periodBoxes.constEnd(); ++iter) {
      if (m_boolManager->value(iter.value()) == true) {
        chosen.append(iter.key());
      }
    }
  }
  return chosen;
 * Ticks the selected periods
 * @param chosenPeriods :: list of selected periods
 */
Anthony Lim's avatar
Anthony Lim committed
void MuonFitPropertyBrowser::setChosenPeriods(
    const QStringList &chosenPeriods) {
  clearChosenPeriods();
  for (const auto &selected : chosenPeriods) {
Anthony Lim's avatar
Anthony Lim committed
    for (auto iter = m_periodBoxes.constBegin();
         iter != m_periodBoxes.constEnd(); ++iter) {
      if (iter.key() == selected) {
        m_boolManager->setValue(iter.value(), true);
      }
    }
  }
 * Ticks the selected periods
 * @param period :: selected periods
 */
void MuonFitPropertyBrowser::setChosenPeriods(const QString &period) {
Anthony Lim's avatar
Anthony Lim committed
  clearChosenPeriods();
  for (auto iter = m_periodBoxes.constBegin(); iter != m_periodBoxes.constEnd();
       ++iter) {
    if (iter.key() == period) {
      m_boolManager->setValue(iter.value(), true);
    }
  }
 * Create a pop up window to select a custom
 * selection of periods
 */
void MuonFitPropertyBrowser::genPeriodWindow() {
Anthony Lim's avatar
Anthony Lim committed
  // reset period window
  m_periodWindow = new QDialog(this);
Anthony Lim's avatar
Anthony Lim committed
  QtGroupPropertyManager *groupManager =
      new QtGroupPropertyManager(m_periodWindow);
  QVBoxLayout *layout = new QVBoxLayout(m_periodWindow);
Anthony Lim's avatar
Anthony Lim committed
  QtTreePropertyBrowser *groupBrowser = new QtTreePropertyBrowser();
  QtProperty *groupSettings = groupManager->addProperty("Period selection");
  for (auto iter = m_periodBoxes.constBegin(); iter != m_periodBoxes.constEnd();
       ++iter) {
    groupSettings->addSubProperty(m_periodBoxes.value(iter.key()));
    m_boolManager->setValue(iter.value(), m_boolManager->value(iter.value()));
  }
  QtCheckBoxFactory *checkBoxFactory = new QtCheckBoxFactory(m_periodWindow);
Anthony Lim's avatar
Anthony Lim committed
  groupBrowser->setFactoryForManager(m_boolManager, checkBoxFactory);
  groupBrowser->addProperty(groupSettings);
  layout->addWidget(groupBrowser);
  m_periodWindow->setLayout(layout);
  m_periodWindow->show();
 * Create a pop up window to create
 * a combination of periods
 */
void MuonFitPropertyBrowser::genCombinePeriodWindow() {
Anthony Lim's avatar
Anthony Lim committed
  // reset combine window
  m_comboWindow = new QDialog(this);
  QVBoxLayout *layout = new QVBoxLayout(m_comboWindow);
Anthony Lim's avatar
Anthony Lim committed
  QFormLayout *formLayout = new QFormLayout;
  m_positiveCombo = new QLineEdit();
  m_negativeCombo = new QLineEdit();
  formLayout->addRow(new QLabel(tr("Combine:")), m_positiveCombo);
  formLayout->addRow(new QLabel(tr("   -    ")), m_negativeCombo);
  layout->addLayout(formLayout);

  QPushButton *applyBtn = new QPushButton("Apply");

  connect(applyBtn, SIGNAL(released()), this, SLOT(combineBtnPressed()));

  layout->addWidget(applyBtn);
  m_comboWindow->setLayout(layout);
  m_comboWindow->show();
 * Get the positive and negative parts of the
 * combination of periods and produce a new
 * tick box. Unticked by default.
 */
void MuonFitPropertyBrowser::combineBtnPressed() {
Anthony Lim's avatar
Anthony Lim committed
  QString value = m_positiveCombo->text();
  if (value.isEmpty()) {
    g_log.error("There are no positive periods (top box)");
    return;
  }
  if (!m_negativeCombo->text().isEmpty()) {
    value.append("-").append(m_negativeCombo->text());
  }
  m_positiveCombo->clear();
  m_negativeCombo->clear();
  addPeriodCheckbox(value);
  int j = m_enumManager->value(m_periodsToFit);
  if (m_periodsToFitOptions[j] == ALL_PERIODS_LABEL) {
    setAllPeriods();
  }
 * sets the label for a single fit and
 * selects the relevant group/pair
 * @param name :: string of the ws
 */
void MuonFitPropertyBrowser::setSingleFitLabel(std::string name) {
Anthony Lim's avatar
Anthony Lim committed
  clearChosenGroups();
  clearChosenPeriods();
  std::vector<std::string> splitName;
  std::string tmpName = name;
  boost::erase_all(tmpName, " ");
  boost::split(splitName, tmpName, boost::is_any_of(";"));
  // set single group/pair
  QString group = QString::fromUtf8(splitName[2].c_str());
  setChosenGroup(group);
  // set period if available
Anthony Lim's avatar
Anthony Lim committed
  if (splitName.size() == 6) {
    QString period = QString::fromUtf8(splitName[4].c_str());
    setChosenPeriods(period);
  }
  setOutputName(name);
  // for single fit in multi fit mode
  if (m_browser->isItemVisible(m_multiFitSettingsGroup)) {
    updateGroupDisplay();
    updatePeriodDisplay();
  }
 * Sets the multifit mode to all groups
 * or all pairs depending on if a  group
 * or pair is selected in the home tab
 * @param isItGroup :: [input] if it is a group (true)
 */
void MuonFitPropertyBrowser::setAllGroupsOrPairs(const bool isItGroup) {
  auto index = m_enumManager->value(m_groupsToFit);
  QString name = m_groupsToFitOptions[index];
  if (name == CUSTOM_LABEL) {
    auto vals = getChosenGroups();
    clearChosenGroups();
    for (const auto &group : vals) {

      for (auto iter = m_groupBoxes.constBegin();
           iter != m_groupBoxes.constEnd(); ++iter) {
        if (iter.key().toStdString() == group.toStdString()) {
          m_boolManager->setValue(iter.value(), true);
        }
      }
    }
  } else if (name == ALL_GROUPS_LABEL) {
Anthony Lim's avatar
Anthony Lim committed
    m_enumManager->setValue(m_groupsToFit, 0);
Anthony Lim's avatar
Anthony Lim committed
    setAllGroups();
    if (getChosenGroups().size() > 0) {
      return;
    }
  } else if (name == ALL_PAIRS_LABEL) { // all pairs is index 1
Anthony Lim's avatar
Anthony Lim committed
    m_enumManager->setValue(m_groupsToFit, 1);
Anthony Lim's avatar
Anthony Lim committed
    setAllPairs();
  if (getChosenGroups().size() > 0) {
    return;
  } else {

    if (isItGroup) {
      // all groups is index 0
      m_enumManager->setValue(m_groupsToFit, 0);
      setAllGroups();
    } else {
      // all pairs is index 1
      m_enumManager->setValue(m_groupsToFit, 1);
      setAllPairs();
    }
  }
Anthony Lim's avatar
Anthony Lim committed
void MuonFitPropertyBrowser::setGroupNames(
    std::vector<std::string> groupNames) {
  m_groupsList = groupNames;
void MuonFitPropertyBrowser::setTFAsymm(bool state) {
Anthony Lim's avatar
Anthony Lim committed
  m_boolManager->setValue(m_TFAsymmMode, state);
} // namespace MantidWidgets
} // namespace MantidQt