Newer
Older
Russell Taylor
committed
#ifndef MANTID_KERNEL_LISTVALIDATOR_H_
#define MANTID_KERNEL_LISTVALIDATOR_H_
#include "MantidKernel/TypedValidator.h"
#include <boost/make_shared.hpp>
#include <vector>
#include <unordered_set>
Russell Taylor
committed
namespace Mantid {
namespace Kernel {
/** ListValidator is a validator that requires the value of a property to be one
of a defined list
of possibilities. The default type is std::string
Russell Taylor
committed
@author Russell Taylor, Tessella Support Services plc
@date 18/06/2008
Copyright © 2008-9 ISIS Rutherford Appleton Laboratory, NScD Oak Ridge
National Laboratory & European Spallation Source
Russell Taylor
committed
This file is part of Mantid.
Mantid is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
Mantid is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
File change history is stored at: <https://github.com/mantidproject/mantid>.
Russell Taylor
committed
Code Documentation is available at: <http://doxygen.mantidproject.org>
*/
template <typename TYPE> class ListValidator : public TypedValidator<TYPE> {
Russell Taylor
committed
public:
/// Default constructor. Sets up an empty list of valid values.
Federico Montesino Pouzols
committed
ListValidator() : TypedValidator<TYPE>(), m_allowMultiSelection(false) {}
* @param values :: A set of values consisting of the valid values
* @param allowMultiSelection :: True if the list allows multi selection
*/
explicit ListValidator(const std::set<TYPE> &values,
const bool allowMultiSelection = false)
: TypedValidator<TYPE>(), m_allowedValues(values.begin(), values.end()),
m_allowMultiSelection(allowMultiSelection) {
if (m_allowMultiSelection) {
throw Kernel::Exception::NotImplementedError(
"The List Validator does not support Multi selection yet");
}
}
/** Constructor
* @param values :: An unordered set of values consisting of the valid values
* @param allowMultiSelection :: True if the list allows multi selection
explicit ListValidator(const std::unordered_set<TYPE> &values,
const bool allowMultiSelection = false)
: TypedValidator<TYPE>(), m_allowedValues(values.begin(), values.end()),
m_allowMultiSelection(allowMultiSelection) {
if (m_allowMultiSelection) {
throw Kernel::Exception::NotImplementedError(
"The List Validator does not support Multi selection yet");
}
}
* @param values :: A vector of the valid values
* @param aliases :: Optional aliases for the valid values.
* @param allowMultiSelection :: True if the list allows multi selection
explicit ListValidator(const std::vector<TYPE> &values,
const std::map<std::string, std::string> &aliases =
std::map<std::string, std::string>(),
const bool allowMultiSelection = false)
: TypedValidator<TYPE>(), m_allowedValues(values.begin(), values.end()),
m_allowMultiSelection(allowMultiSelection) {
for (auto aliasIt = m_aliases.begin(); aliasIt != m_aliases.end();
++aliasIt) {
if (values.end() ==
std::find(values.begin(), values.end(),
boost::lexical_cast<TYPE>(aliasIt->second))) {
throw std::invalid_argument("Alias " + aliasIt->first +
if (m_allowMultiSelection) {
throw Kernel::Exception::NotImplementedError(
"The List Validator does not support Multi selection yet");
}
/// Clone the validator
IValidator_sptr clone() const override {
return boost::make_shared<ListValidator<TYPE>>(*this);
}
/**
* Returns the set of allowed values currently defined
* @returns A set of allowed values that this validator will currently allow
*/
std::vector<std::string> allowedValues() const override {
/// The interface requires strings
std::vector<std::string> allowedStrings;
allowedStrings.reserve(m_allowedValues.size());
auto cend = m_allowedValues.end();
for (auto cit = m_allowedValues.begin(); cit != cend; ++cit) {
allowedStrings.push_back(boost::lexical_cast<std::string>(*cit));
}
return allowedStrings;
}
* Add value to the list of allowable values if it's not already there
* @param value :: A value of the templated type
*/
void addAllowedValue(const TYPE &value) {
// add only new values
if (std::find(m_allowedValues.begin(), m_allowedValues.end(), value) ==
m_allowedValues.end()) {
m_allowedValues.push_back(value);
}
/**
* Return an allowed value (as a string) given an alias.
* @param alias :: An alias string.
* @return :: Allowed value or throw if alias is unknown.
*/
std::string getValueForAlias(const std::string &alias) const override {
auto aliasIt = m_aliases.find(alias);
if (aliasIt == m_aliases.end()) {
throw std::invalid_argument("Unknown alias found " + alias);
}
return aliasIt->second;
}
bool isMultipleSelectionAllowed() override { return m_allowMultiSelection; }
void setMultipleSelectionAllowed(const bool isMultiSelectionAllowed) {
if (isMultiSelectionAllowed) {
throw Kernel::Exception::NotImplementedError(
"The List Validator does not support Multi selection yet");
}
m_allowMultiSelection = isMultiSelectionAllowed;
/** Checks if the string passed is in the list
* @param value :: The value to test
* @return "" if the value is on the list, or "The value is not in the list
* of allowed values"
std::string checkValidity(const TYPE &value) const override {
if (m_allowedValues.end() !=
std::find(m_allowedValues.begin(), m_allowedValues.end(), value)) {
} else {
if (isEmpty(value))
return "Select a value";
if (isAlias(value))
return "_alias";
std::ostringstream os;
os << "The value \"" << value
<< "\" is not in the list of allowed values";
return os.str();
Russell Taylor
committed
/**
* Is the value considered empty.
* @param value :: The value to check
* @return True if it is considered empty
*/
template <typename T> bool isEmpty(const T &value) const {
UNUSED_ARG(value) return false;
}
/**
* Is the value considered empty. Specialized string version to use empty
* @param value :: The value to check
* @return True if it is considered empty
*/
bool isEmpty(const std::string &value) const { return value.empty(); }
/**
* Test if a value is an alias of an alowed value.
* @param value :: Value to test.
* @return :: True if it's an alias.
*/
template <typename T> bool isAlias(const T &value) const {
std::string strValue = boost::lexical_cast<std::string>(value);
return m_aliases.find(strValue) != m_aliases.end();
}
/**
* Test if a value is an alias of an alowed value.
* @param value :: Value to test.
* @return :: True if it's an alias.
*/
bool isAlias(const std::string &value) const {
return m_aliases.find(value) != m_aliases.end();
/// The set of valid values
std::vector<TYPE> m_allowedValues;
/// The optional aliases for the allowed values.
std::map<std::string, std::string> m_aliases;
/// if the validator should allow multiple selection
Federico Montesino Pouzols
committed
bool m_allowMultiSelection;
Russell Taylor
committed
};
/// ListValidator<std::string> is used heavily
typedef ListValidator<std::string> StringListValidator;
Russell Taylor
committed
} // namespace Kernel
} // namespace Mantid
#endif /*MANTID_KERNEL_LISTVALIDATOR_H_*/