2013-12-07 15:14:30 +00:00
|
|
|
#ifndef slic3r_Config_hpp_
|
|
|
|
#define slic3r_Config_hpp_
|
|
|
|
|
2016-12-12 14:56:42 +00:00
|
|
|
#include <assert.h>
|
2013-12-07 15:14:30 +00:00
|
|
|
#include <map>
|
2013-12-21 23:39:03 +00:00
|
|
|
#include <climits>
|
2013-12-20 19:54:11 +00:00
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
2013-12-21 09:46:43 +00:00
|
|
|
#include <iostream>
|
2014-01-04 23:36:33 +00:00
|
|
|
#include <stdexcept>
|
2013-12-07 15:14:30 +00:00
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
2015-12-07 23:39:54 +00:00
|
|
|
#include "libslic3r.h"
|
2013-12-20 19:54:11 +00:00
|
|
|
#include "Point.hpp"
|
2013-12-07 15:14:30 +00:00
|
|
|
|
2017-11-01 18:30:05 +00:00
|
|
|
#include <boost/property_tree/ptree.hpp>
|
|
|
|
|
2013-12-20 15:37:28 +00:00
|
|
|
namespace Slic3r {
|
2013-12-07 15:14:30 +00:00
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Name of the configuration option.
|
2017-10-17 14:01:18 +00:00
|
|
|
typedef std::string t_config_option_key;
|
|
|
|
typedef std::vector<std::string> t_config_option_keys;
|
2013-12-07 15:14:30 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
extern std::string escape_string_cstyle(const std::string &str);
|
|
|
|
extern std::string escape_strings_cstyle(const std::vector<std::string> &strs);
|
|
|
|
extern bool unescape_string_cstyle(const std::string &str, std::string &out);
|
|
|
|
extern bool unescape_strings_cstyle(const std::string &str, std::vector<std::string> &out);
|
|
|
|
|
|
|
|
|
|
|
|
// Type of a configuration value.
|
|
|
|
enum ConfigOptionType {
|
2017-10-25 10:53:31 +00:00
|
|
|
coVectorType = 0x4000,
|
|
|
|
coNone = 0,
|
2017-10-17 14:01:18 +00:00
|
|
|
// single float
|
2017-10-25 10:53:31 +00:00
|
|
|
coFloat = 1,
|
2017-10-17 14:01:18 +00:00
|
|
|
// vector of floats
|
2017-10-25 10:53:31 +00:00
|
|
|
coFloats = coFloat + coVectorType,
|
2017-10-17 14:01:18 +00:00
|
|
|
// single int
|
2017-10-25 10:53:31 +00:00
|
|
|
coInt = 2,
|
2017-10-17 14:01:18 +00:00
|
|
|
// vector of ints
|
2017-10-25 10:53:31 +00:00
|
|
|
coInts = coInt + coVectorType,
|
2017-10-17 14:01:18 +00:00
|
|
|
// single string
|
2017-10-25 10:53:31 +00:00
|
|
|
coString = 3,
|
2017-10-17 14:01:18 +00:00
|
|
|
// vector of strings
|
2017-10-25 10:53:31 +00:00
|
|
|
coStrings = coString + coVectorType,
|
2017-10-17 14:01:18 +00:00
|
|
|
// percent value. Currently only used for infill.
|
2017-10-25 10:53:31 +00:00
|
|
|
coPercent = 4,
|
2017-10-17 14:01:18 +00:00
|
|
|
// percents value. Currently used for retract before wipe only.
|
2017-10-25 10:53:31 +00:00
|
|
|
coPercents = coPercent + coVectorType,
|
2017-10-17 14:01:18 +00:00
|
|
|
// a fraction or an absolute value
|
2017-10-25 10:53:31 +00:00
|
|
|
coFloatOrPercent = 5,
|
2017-10-17 14:01:18 +00:00
|
|
|
// single 2d point. Currently not used.
|
2017-10-25 10:53:31 +00:00
|
|
|
coPoint = 6,
|
2017-10-17 14:01:18 +00:00
|
|
|
// vector of 2d points. Currently used for the definition of the print bed and for the extruder offsets.
|
2017-10-25 10:53:31 +00:00
|
|
|
coPoints = coPoint + coVectorType,
|
2017-10-17 14:01:18 +00:00
|
|
|
// single boolean value
|
2017-10-25 10:53:31 +00:00
|
|
|
coBool = 7,
|
2017-10-17 14:01:18 +00:00
|
|
|
// vector of boolean values
|
2017-10-25 10:53:31 +00:00
|
|
|
coBools = coBool + coVectorType,
|
2017-10-17 14:01:18 +00:00
|
|
|
// a generic enum
|
2017-10-25 10:53:31 +00:00
|
|
|
coEnum = 8,
|
2017-10-17 14:01:18 +00:00
|
|
|
};
|
2016-11-01 12:41:24 +00:00
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// A generic value of a configuration option.
|
2013-12-20 15:37:28 +00:00
|
|
|
class ConfigOption {
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
virtual ~ConfigOption() {}
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
virtual ConfigOptionType type() const = 0;
|
|
|
|
virtual std::string serialize() const = 0;
|
|
|
|
virtual bool deserialize(const std::string &str, bool append = false) = 0;
|
|
|
|
virtual ConfigOption* clone() const = 0;
|
|
|
|
// Set a value from a ConfigOption. The two options should be compatible.
|
|
|
|
virtual void set(const ConfigOption *option) = 0;
|
|
|
|
virtual int getInt() const { throw std::runtime_error("Calling ConfigOption::getInt on a non-int ConfigOption"); return 0; }
|
|
|
|
virtual double getFloat() const { throw std::runtime_error("Calling ConfigOption::getFloat on a non-float ConfigOption"); return 0; }
|
|
|
|
virtual bool getBool() const { throw std::runtime_error("Calling ConfigOption::getBool on a non-boolean ConfigOption"); return 0; }
|
|
|
|
virtual void setInt(int /* val */) { throw std::runtime_error("Calling ConfigOption::setInt on a non-int ConfigOption"); }
|
|
|
|
virtual bool operator==(const ConfigOption &rhs) const = 0;
|
|
|
|
bool operator!=(const ConfigOption &rhs) const { return ! (*this == rhs); }
|
2017-10-25 10:53:31 +00:00
|
|
|
bool is_scalar() const { return (int(this->type()) & int(coVectorType)) == 0; }
|
|
|
|
bool is_vector() const { return ! this->is_scalar(); }
|
2013-12-20 15:37:28 +00:00
|
|
|
};
|
|
|
|
|
2017-11-17 17:46:03 +00:00
|
|
|
typedef ConfigOption* ConfigOptionPtr;
|
|
|
|
typedef const ConfigOption* ConfigOptionConstPtr;
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Value of a single valued option (bool, int, float, string, point, enum)
|
2015-12-07 18:39:49 +00:00
|
|
|
template <class T>
|
|
|
|
class ConfigOptionSingle : public ConfigOption {
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2015-12-07 18:39:49 +00:00
|
|
|
T value;
|
2017-10-17 14:01:18 +00:00
|
|
|
explicit ConfigOptionSingle(T value) : value(value) {}
|
|
|
|
operator T() const { return this->value; }
|
|
|
|
|
|
|
|
void set(const ConfigOption *rhs) override
|
|
|
|
{
|
|
|
|
if (rhs->type() != this->type())
|
|
|
|
throw std::runtime_error("ConfigOptionSingle: Assigning an incompatible type");
|
|
|
|
assert(dynamic_cast<const ConfigOptionSingle<T>*>(rhs));
|
|
|
|
this->value = static_cast<const ConfigOptionSingle<T>*>(rhs)->value;
|
2017-11-01 18:30:05 +00:00
|
|
|
}
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
bool operator==(const ConfigOption &rhs) const override
|
|
|
|
{
|
|
|
|
if (rhs.type() != this->type())
|
|
|
|
throw std::runtime_error("ConfigOptionSingle: Comparing incompatible types");
|
|
|
|
assert(dynamic_cast<const ConfigOptionSingle<T>*>(&rhs));
|
|
|
|
return this->value == static_cast<const ConfigOptionSingle<T>*>(&rhs)->value;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(const T &rhs) const { return this->value == rhs; }
|
|
|
|
bool operator!=(const T &rhs) const { return this->value != rhs; }
|
2015-12-07 18:39:49 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Value of a vector valued option (bools, ints, floats, strings, points)
|
2015-05-02 19:43:22 +00:00
|
|
|
class ConfigOptionVectorBase : public ConfigOption {
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-05-19 17:24:21 +00:00
|
|
|
// Currently used only to initialize the PlaceholderParser.
|
2015-05-02 19:43:22 +00:00
|
|
|
virtual std::vector<std::string> vserialize() const = 0;
|
2017-10-25 10:53:31 +00:00
|
|
|
// Set from a vector of ConfigOptions.
|
|
|
|
// If the rhs ConfigOption is scalar, then its value is used,
|
|
|
|
// otherwise for each of rhs, the first value of a vector is used.
|
|
|
|
// This function is useful to collect values for multiple extrder / filament settings.
|
|
|
|
virtual void set(const std::vector<const ConfigOption*> &rhs) = 0;
|
|
|
|
// Set a single vector item from either a scalar option or the first value of a vector option.vector of ConfigOptions.
|
|
|
|
// This function is useful to split values from multiple extrder / filament settings into separate configurations.
|
|
|
|
virtual void set_at(const ConfigOption *rhs, size_t i, size_t j) = 0;
|
|
|
|
|
2017-11-01 18:30:05 +00:00
|
|
|
virtual void resize(size_t n, const ConfigOption *opt_default = nullptr) = 0;
|
|
|
|
|
2017-11-17 10:15:46 +00:00
|
|
|
// Get size of this vector.
|
|
|
|
virtual size_t size() const = 0;
|
|
|
|
// Is this vector empty?
|
|
|
|
virtual bool empty() const = 0;
|
2017-11-01 18:30:05 +00:00
|
|
|
|
2017-10-25 10:53:31 +00:00
|
|
|
protected:
|
|
|
|
// Used to verify type compatibility when assigning to / from a scalar ConfigOption.
|
|
|
|
ConfigOptionType scalar_type() const { return static_cast<ConfigOptionType>(this->type() - coVectorType); }
|
2015-05-02 19:43:22 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Value of a vector valued option (bools, ints, floats, strings, points), template
|
2014-01-04 23:36:33 +00:00
|
|
|
template <class T>
|
2015-05-02 19:43:22 +00:00
|
|
|
class ConfigOptionVector : public ConfigOptionVectorBase
|
2014-01-04 23:36:33 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionVector() {}
|
|
|
|
explicit ConfigOptionVector(size_t n, const T &value) : values(n, value) {}
|
|
|
|
explicit ConfigOptionVector(std::initializer_list<T> il) : values(std::move(il)) {}
|
2017-11-17 10:15:46 +00:00
|
|
|
explicit ConfigOptionVector(const std::vector<T> &values) : values(values) {}
|
|
|
|
explicit ConfigOptionVector(std::vector<T> &&values) : values(std::move(values)) {}
|
2014-01-04 23:36:33 +00:00
|
|
|
std::vector<T> values;
|
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
void set(const ConfigOption *rhs) override
|
|
|
|
{
|
|
|
|
if (rhs->type() != this->type())
|
|
|
|
throw std::runtime_error("ConfigOptionVector: Assigning an incompatible type");
|
|
|
|
assert(dynamic_cast<const ConfigOptionVector<T>*>(rhs));
|
|
|
|
this->values = static_cast<const ConfigOptionVector<T>*>(rhs)->values;
|
2017-10-25 10:53:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Set from a vector of ConfigOptions.
|
|
|
|
// If the rhs ConfigOption is scalar, then its value is used,
|
|
|
|
// otherwise for each of rhs, the first value of a vector is used.
|
|
|
|
// This function is useful to collect values for multiple extrder / filament settings.
|
|
|
|
void set(const std::vector<const ConfigOption*> &rhs) override
|
|
|
|
{
|
|
|
|
this->values.clear();
|
|
|
|
this->values.reserve(rhs.size());
|
|
|
|
for (const ConfigOption *opt : rhs) {
|
|
|
|
if (opt->type() == this->type()) {
|
|
|
|
auto other = static_cast<const ConfigOptionVector<T>*>(opt);
|
|
|
|
if (other->values.empty())
|
|
|
|
throw std::runtime_error("ConfigOptionVector::set(): Assigning from an empty vector");
|
|
|
|
this->values.emplace_back(other->values.front());
|
|
|
|
} else if (opt->type() == this->scalar_type())
|
|
|
|
this->values.emplace_back(static_cast<const ConfigOptionSingle<T>*>(opt)->value);
|
|
|
|
else
|
|
|
|
throw std::runtime_error("ConfigOptionVector::set():: Assigning an incompatible type");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set a single vector item from either a scalar option or the first value of a vector option.vector of ConfigOptions.
|
|
|
|
// This function is useful to split values from multiple extrder / filament settings into separate configurations.
|
|
|
|
void set_at(const ConfigOption *rhs, size_t i, size_t j) override
|
|
|
|
{
|
|
|
|
// It is expected that the vector value has at least one value, which is the default, if not overwritten.
|
|
|
|
assert(! this->values.empty());
|
|
|
|
if (this->values.size() <= i) {
|
|
|
|
// Resize this vector, fill in the new vector fields with the copy of the first field.
|
|
|
|
T v = this->values.front();
|
|
|
|
this->values.resize(i + 1, v);
|
|
|
|
}
|
|
|
|
if (rhs->type() == this->type()) {
|
|
|
|
// Assign the first value of the rhs vector.
|
|
|
|
auto other = static_cast<const ConfigOptionVector<T>*>(rhs);
|
|
|
|
if (other->values.empty())
|
|
|
|
throw std::runtime_error("ConfigOptionVector::set_at(): Assigning from an empty vector");
|
|
|
|
this->values[i] = other->get_at(j);
|
|
|
|
} else if (rhs->type() == this->scalar_type())
|
|
|
|
this->values[i] = static_cast<const ConfigOptionSingle<T>*>(rhs)->value;
|
|
|
|
else
|
|
|
|
throw std::runtime_error("ConfigOptionVector::set_at(): Assigning an incompatible type");
|
|
|
|
}
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
T& get_at(size_t i)
|
|
|
|
{
|
2016-12-12 14:56:42 +00:00
|
|
|
assert(! this->values.empty());
|
|
|
|
return (i < this->values.size()) ? this->values[i] : this->values.front();
|
2017-10-25 10:53:31 +00:00
|
|
|
}
|
2017-09-19 11:55:48 +00:00
|
|
|
|
|
|
|
const T& get_at(size_t i) const { return const_cast<ConfigOptionVector<T>*>(this)->get_at(i); }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
2017-11-01 18:30:05 +00:00
|
|
|
// Resize this vector by duplicating the last value.
|
|
|
|
// If the current vector is empty, the default value is used instead.
|
|
|
|
void resize(size_t n, const ConfigOption *opt_default = nullptr) override
|
|
|
|
{
|
|
|
|
assert(opt_default == nullptr || opt_default->is_vector());
|
2017-11-02 15:21:34 +00:00
|
|
|
// assert(opt_default == nullptr || dynamic_cast<ConfigOptionVector<T>>(opt_default));
|
2017-11-01 18:30:05 +00:00
|
|
|
assert(! this->values.empty() || opt_default != nullptr);
|
|
|
|
if (n == 0)
|
|
|
|
this->values.clear();
|
|
|
|
else if (n < this->values.size())
|
|
|
|
this->values.erase(this->values.begin() + n, this->values.end());
|
2017-11-02 15:21:34 +00:00
|
|
|
else if (n > this->values.size()) {
|
2017-11-01 18:30:05 +00:00
|
|
|
if (this->values.empty()) {
|
2017-11-02 15:21:34 +00:00
|
|
|
if (opt_default == nullptr)
|
2017-11-01 18:30:05 +00:00
|
|
|
throw std::runtime_error("ConfigOptionVector::resize(): No default value provided.");
|
|
|
|
if (opt_default->type() != this->type())
|
|
|
|
throw std::runtime_error("ConfigOptionVector::resize(): Extending with an incompatible type.");
|
|
|
|
this->values.resize(n, static_cast<const ConfigOptionVector<T>*>(opt_default)->values.front());
|
|
|
|
} else {
|
|
|
|
// Resize by duplicating the last value.
|
|
|
|
this->values.resize(n, this->values.back());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-17 10:15:46 +00:00
|
|
|
size_t size() const override { return this->values.size(); }
|
|
|
|
bool empty() const override { return this->values.empty(); }
|
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool operator==(const ConfigOption &rhs) const override
|
|
|
|
{
|
|
|
|
if (rhs.type() != this->type())
|
|
|
|
throw std::runtime_error("ConfigOptionVector: Comparing incompatible types");
|
|
|
|
assert(dynamic_cast<const ConfigOptionVector<T>*>(&rhs));
|
|
|
|
return this->values == static_cast<const ConfigOptionVector<T>*>(&rhs)->values;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(const std::vector<T> &rhs) const { return this->values == rhs; }
|
|
|
|
bool operator!=(const std::vector<T> &rhs) const { return this->values != rhs; }
|
2014-01-04 23:36:33 +00:00
|
|
|
};
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionFloat : public ConfigOptionSingle<double>
|
2013-12-07 15:14:30 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionFloat() : ConfigOptionSingle<double>(0) {}
|
|
|
|
explicit ConfigOptionFloat(double _value) : ConfigOptionSingle<double>(_value) {}
|
2017-10-17 14:01:18 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coFloat; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
double getFloat() const override { return this->value; }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionFloat(*this); }
|
|
|
|
bool operator==(const ConfigOptionFloat &rhs) const { return this->value == rhs.value; }
|
2013-12-20 19:54:11 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2013-12-20 19:54:11 +00:00
|
|
|
std::ostringstream ss;
|
|
|
|
ss << this->value;
|
|
|
|
return ss.str();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-20 19:54:11 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-07-27 08:39:43 +00:00
|
|
|
UNUSED(append);
|
2014-08-04 09:34:53 +00:00
|
|
|
std::istringstream iss(str);
|
2016-03-13 14:25:50 +00:00
|
|
|
iss >> this->value;
|
|
|
|
return !iss.fail();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ConfigOptionFloat& operator=(const ConfigOption *opt)
|
|
|
|
{
|
|
|
|
this->set(opt);
|
|
|
|
return *this;
|
|
|
|
}
|
2013-12-20 15:37:28 +00:00
|
|
|
};
|
|
|
|
|
2015-05-02 19:43:22 +00:00
|
|
|
class ConfigOptionFloats : public ConfigOptionVector<double>
|
2013-12-21 13:27:58 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionFloats() : ConfigOptionVector<double>() {}
|
|
|
|
explicit ConfigOptionFloats(size_t n, double value) : ConfigOptionVector<double>(n, value) {}
|
|
|
|
explicit ConfigOptionFloats(std::initializer_list<double> il) : ConfigOptionVector<double>(std::move(il)) {}
|
2018-06-20 11:57:37 +00:00
|
|
|
explicit ConfigOptionFloats(const std::vector<double> &vec) : ConfigOptionVector<double>(vec) {}
|
|
|
|
explicit ConfigOptionFloats(std::vector<double> &&vec) : ConfigOptionVector<double>(std::move(vec)) {}
|
2017-11-01 18:30:05 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coFloats; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionFloats(*this); }
|
|
|
|
bool operator==(const ConfigOptionFloats &rhs) const { return this->values == rhs.values; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2013-12-21 13:27:58 +00:00
|
|
|
std::ostringstream ss;
|
2013-12-22 00:38:10 +00:00
|
|
|
for (std::vector<double>::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
|
2013-12-21 13:27:58 +00:00
|
|
|
if (it - this->values.begin() != 0) ss << ",";
|
|
|
|
ss << *it;
|
|
|
|
}
|
|
|
|
return ss.str();
|
2017-11-01 18:30:05 +00:00
|
|
|
}
|
2013-12-21 13:27:58 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::vector<std::string> vserialize() const override
|
|
|
|
{
|
2015-05-02 19:43:22 +00:00
|
|
|
std::vector<std::string> vv;
|
2016-11-01 12:41:24 +00:00
|
|
|
vv.reserve(this->values.size());
|
2015-05-02 19:43:22 +00:00
|
|
|
for (std::vector<double>::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
|
|
|
|
std::ostringstream ss;
|
|
|
|
ss << *it;
|
|
|
|
vv.push_back(ss.str());
|
|
|
|
}
|
|
|
|
return vv;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2015-05-02 19:43:22 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-06-14 15:51:14 +00:00
|
|
|
if (! append)
|
|
|
|
this->values.clear();
|
2013-12-21 13:27:58 +00:00
|
|
|
std::istringstream is(str);
|
|
|
|
std::string item_str;
|
|
|
|
while (std::getline(is, item_str, ',')) {
|
2014-08-04 09:34:53 +00:00
|
|
|
std::istringstream iss(item_str);
|
|
|
|
double value;
|
|
|
|
iss >> value;
|
|
|
|
this->values.push_back(value);
|
2013-12-21 13:27:58 +00:00
|
|
|
}
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ConfigOptionFloats& operator=(const ConfigOption *opt)
|
|
|
|
{
|
|
|
|
this->set(opt);
|
|
|
|
return *this;
|
|
|
|
}
|
2013-12-21 13:27:58 +00:00
|
|
|
};
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionInt : public ConfigOptionSingle<int>
|
2013-12-20 15:37:28 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionInt() : ConfigOptionSingle<int>(0) {}
|
|
|
|
explicit ConfigOptionInt(int value) : ConfigOptionSingle<int>(value) {}
|
|
|
|
explicit ConfigOptionInt(double _value) : ConfigOptionSingle<int>(int(floor(_value + 0.5))) {}
|
2013-12-20 19:54:11 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coInt; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
int getInt() const override { return this->value; }
|
|
|
|
void setInt(int val) { this->value = val; }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionInt(*this); }
|
|
|
|
bool operator==(const ConfigOptionInt &rhs) const { return this->value == rhs.value; }
|
2013-12-20 19:54:11 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2013-12-20 19:54:11 +00:00
|
|
|
std::ostringstream ss;
|
|
|
|
ss << this->value;
|
|
|
|
return ss.str();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-20 19:54:11 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-07-27 08:39:43 +00:00
|
|
|
UNUSED(append);
|
2014-08-04 09:34:53 +00:00
|
|
|
std::istringstream iss(str);
|
2016-03-13 14:25:50 +00:00
|
|
|
iss >> this->value;
|
|
|
|
return !iss.fail();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ConfigOptionInt& operator=(const ConfigOption *opt)
|
|
|
|
{
|
|
|
|
this->set(opt);
|
|
|
|
return *this;
|
|
|
|
}
|
2013-12-20 15:37:28 +00:00
|
|
|
};
|
|
|
|
|
2015-05-02 19:43:22 +00:00
|
|
|
class ConfigOptionInts : public ConfigOptionVector<int>
|
2013-12-21 13:27:58 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionInts() : ConfigOptionVector<int>() {}
|
|
|
|
explicit ConfigOptionInts(size_t n, int value) : ConfigOptionVector<int>(n, value) {}
|
|
|
|
explicit ConfigOptionInts(std::initializer_list<int> il) : ConfigOptionVector<int>(std::move(il)) {}
|
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coInts; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionInts(*this); }
|
|
|
|
ConfigOptionInts& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionInts &rhs) const { return this->values == rhs.values; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
std::string serialize() const override {
|
2013-12-21 13:27:58 +00:00
|
|
|
std::ostringstream ss;
|
|
|
|
for (std::vector<int>::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
|
|
|
|
if (it - this->values.begin() != 0) ss << ",";
|
|
|
|
ss << *it;
|
|
|
|
}
|
|
|
|
return ss.str();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-21 13:27:58 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::vector<std::string> vserialize() const override
|
|
|
|
{
|
2015-05-02 19:43:22 +00:00
|
|
|
std::vector<std::string> vv;
|
2016-11-01 12:41:24 +00:00
|
|
|
vv.reserve(this->values.size());
|
2015-05-02 19:43:22 +00:00
|
|
|
for (std::vector<int>::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
|
|
|
|
std::ostringstream ss;
|
|
|
|
ss << *it;
|
|
|
|
vv.push_back(ss.str());
|
|
|
|
}
|
|
|
|
return vv;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2015-05-02 19:43:22 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-06-14 15:51:14 +00:00
|
|
|
if (! append)
|
|
|
|
this->values.clear();
|
2013-12-21 13:27:58 +00:00
|
|
|
std::istringstream is(str);
|
|
|
|
std::string item_str;
|
|
|
|
while (std::getline(is, item_str, ',')) {
|
2014-08-04 09:34:53 +00:00
|
|
|
std::istringstream iss(item_str);
|
|
|
|
int value;
|
|
|
|
iss >> value;
|
|
|
|
this->values.push_back(value);
|
2013-12-21 13:27:58 +00:00
|
|
|
}
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-21 13:27:58 +00:00
|
|
|
};
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionString : public ConfigOptionSingle<std::string>
|
2013-12-20 15:37:28 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionString() : ConfigOptionSingle<std::string>("") {}
|
|
|
|
explicit ConfigOptionString(const std::string &value) : ConfigOptionSingle<std::string>(value) {}
|
2013-12-20 19:54:11 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coString; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionString(*this); }
|
|
|
|
ConfigOptionString& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionString &rhs) const { return this->value == rhs.value; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
|
|
|
return escape_string_cstyle(this->value);
|
2016-11-01 12:41:24 +00:00
|
|
|
}
|
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-07-27 08:39:43 +00:00
|
|
|
UNUSED(append);
|
2016-11-01 12:41:24 +00:00
|
|
|
return unescape_string_cstyle(str, this->value);
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-20 15:37:28 +00:00
|
|
|
};
|
|
|
|
|
2013-12-21 23:39:03 +00:00
|
|
|
// semicolon-separated strings
|
2015-05-02 19:43:22 +00:00
|
|
|
class ConfigOptionStrings : public ConfigOptionVector<std::string>
|
2013-12-21 23:39:03 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionStrings() : ConfigOptionVector<std::string>() {}
|
|
|
|
explicit ConfigOptionStrings(size_t n, const std::string &value) : ConfigOptionVector<std::string>(n, value) {}
|
2017-11-17 10:15:46 +00:00
|
|
|
explicit ConfigOptionStrings(const std::vector<std::string> &values) : ConfigOptionVector<std::string>(values) {}
|
|
|
|
explicit ConfigOptionStrings(std::vector<std::string> &&values) : ConfigOptionVector<std::string>(std::move(values)) {}
|
2017-11-01 18:30:05 +00:00
|
|
|
explicit ConfigOptionStrings(std::initializer_list<std::string> il) : ConfigOptionVector<std::string>(std::move(il)) {}
|
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coStrings; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionStrings(*this); }
|
|
|
|
ConfigOptionStrings& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionStrings &rhs) const { return this->values == rhs.values; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2016-11-01 12:41:24 +00:00
|
|
|
return escape_strings_cstyle(this->values);
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-21 23:39:03 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::vector<std::string> vserialize() const override
|
|
|
|
{
|
2015-05-02 19:43:22 +00:00
|
|
|
return this->values;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2015-05-02 19:43:22 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-06-14 15:51:14 +00:00
|
|
|
if (! append)
|
|
|
|
this->values.clear();
|
2016-11-01 12:41:24 +00:00
|
|
|
return unescape_strings_cstyle(str, this->values);
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-21 23:39:03 +00:00
|
|
|
};
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionPercent : public ConfigOptionFloat
|
2014-03-22 15:23:33 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionPercent() : ConfigOptionFloat(0) {}
|
|
|
|
explicit ConfigOptionPercent(double _value) : ConfigOptionFloat(_value) {}
|
2014-03-22 15:23:33 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coPercent; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionPercent(*this); }
|
|
|
|
ConfigOptionPercent& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionPercent &rhs) const { return this->value == rhs.value; }
|
|
|
|
double get_abs_value(double ratio_over) const { return ratio_over * this->value / 100; }
|
2014-03-22 15:23:33 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2014-03-22 15:23:33 +00:00
|
|
|
std::ostringstream ss;
|
|
|
|
ss << this->value;
|
|
|
|
std::string s(ss.str());
|
|
|
|
s += "%";
|
|
|
|
return s;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2014-03-22 15:23:33 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-07-27 08:39:43 +00:00
|
|
|
UNUSED(append);
|
2014-03-22 15:23:33 +00:00
|
|
|
// don't try to parse the trailing % since it's optional
|
2014-08-04 09:34:53 +00:00
|
|
|
std::istringstream iss(str);
|
2016-03-13 14:25:50 +00:00
|
|
|
iss >> this->value;
|
|
|
|
return !iss.fail();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2014-03-22 15:23:33 +00:00
|
|
|
};
|
|
|
|
|
2017-05-19 17:24:21 +00:00
|
|
|
class ConfigOptionPercents : public ConfigOptionFloats
|
|
|
|
{
|
2017-10-17 14:01:18 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionPercents() : ConfigOptionFloats() {}
|
|
|
|
explicit ConfigOptionPercents(size_t n, double value) : ConfigOptionFloats(n, value) {}
|
|
|
|
explicit ConfigOptionPercents(std::initializer_list<double> il) : ConfigOptionFloats(std::move(il)) {}
|
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coPercents; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionPercents(*this); }
|
|
|
|
ConfigOptionPercents& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionPercents &rhs) const { return this->values == rhs.values; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2017-05-19 17:24:21 +00:00
|
|
|
std::ostringstream ss;
|
2017-05-30 19:29:43 +00:00
|
|
|
for (const auto &v : this->values) {
|
2017-05-19 17:24:21 +00:00
|
|
|
if (&v != &this->values.front()) ss << ",";
|
|
|
|
ss << v << "%";
|
|
|
|
}
|
|
|
|
std::string str = ss.str();
|
|
|
|
return str;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2017-05-19 17:24:21 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::vector<std::string> vserialize() const override
|
|
|
|
{
|
2017-05-19 17:24:21 +00:00
|
|
|
std::vector<std::string> vv;
|
|
|
|
vv.reserve(this->values.size());
|
|
|
|
for (const auto v : this->values) {
|
|
|
|
std::ostringstream ss;
|
|
|
|
ss << v;
|
|
|
|
std::string sout = ss.str() + "%";
|
|
|
|
vv.push_back(sout);
|
|
|
|
}
|
|
|
|
return vv;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2017-05-19 17:24:21 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-06-14 15:51:14 +00:00
|
|
|
if (! append)
|
|
|
|
this->values.clear();
|
2017-05-19 17:24:21 +00:00
|
|
|
std::istringstream is(str);
|
|
|
|
std::string item_str;
|
|
|
|
while (std::getline(is, item_str, ',')) {
|
|
|
|
std::istringstream iss(item_str);
|
|
|
|
double value;
|
|
|
|
// don't try to parse the trailing % since it's optional
|
|
|
|
iss >> value;
|
|
|
|
this->values.push_back(value);
|
|
|
|
}
|
|
|
|
return true;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2017-05-19 17:24:21 +00:00
|
|
|
};
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionFloatOrPercent : public ConfigOptionPercent
|
2013-12-20 15:37:28 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2013-12-20 15:37:28 +00:00
|
|
|
bool percent;
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionFloatOrPercent() : ConfigOptionPercent(0), percent(false) {}
|
|
|
|
explicit ConfigOptionFloatOrPercent(double _value, bool _percent) : ConfigOptionPercent(_value), percent(_percent) {}
|
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coFloatOrPercent; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
2017-10-17 14:01:18 +00:00
|
|
|
ConfigOption* clone() const override { return new ConfigOptionFloatOrPercent(*this); }
|
|
|
|
ConfigOptionFloatOrPercent& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
2018-02-12 18:06:05 +00:00
|
|
|
bool operator==(const ConfigOption &rhs) const override
|
|
|
|
{
|
|
|
|
if (rhs.type() != this->type())
|
|
|
|
throw std::runtime_error("ConfigOptionFloatOrPercent: Comparing incompatible types");
|
|
|
|
assert(dynamic_cast<const ConfigOptionFloatOrPercent*>(&rhs));
|
|
|
|
return *this == *static_cast<const ConfigOptionFloatOrPercent*>(&rhs);
|
|
|
|
}
|
2017-10-17 14:01:18 +00:00
|
|
|
bool operator==(const ConfigOptionFloatOrPercent &rhs) const
|
|
|
|
{ return this->value == rhs.value && this->percent == rhs.percent; }
|
|
|
|
double get_abs_value(double ratio_over) const
|
|
|
|
{ return this->percent ? (ratio_over * this->value / 100) : this->value; }
|
|
|
|
|
|
|
|
void set(const ConfigOption *rhs) override {
|
|
|
|
if (rhs->type() != this->type())
|
|
|
|
throw std::runtime_error("ConfigOptionFloatOrPercent: Assigning an incompatible type");
|
|
|
|
assert(dynamic_cast<const ConfigOptionFloatOrPercent*>(rhs));
|
|
|
|
*this = *static_cast<const ConfigOptionFloatOrPercent*>(rhs);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2013-12-20 19:54:11 +00:00
|
|
|
std::ostringstream ss;
|
|
|
|
ss << this->value;
|
|
|
|
std::string s(ss.str());
|
|
|
|
if (this->percent) s += "%";
|
|
|
|
return s;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-20 19:54:11 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-07-27 08:39:43 +00:00
|
|
|
UNUSED(append);
|
2014-08-04 09:34:53 +00:00
|
|
|
this->percent = str.find_first_of("%") != std::string::npos;
|
|
|
|
std::istringstream iss(str);
|
2016-03-13 14:25:50 +00:00
|
|
|
iss >> this->value;
|
|
|
|
return !iss.fail();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-20 19:54:11 +00:00
|
|
|
};
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionPoint : public ConfigOptionSingle<Pointf>
|
2013-12-20 19:54:11 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionPoint() : ConfigOptionSingle<Pointf>(Pointf(0,0)) {}
|
|
|
|
explicit ConfigOptionPoint(const Pointf &value) : ConfigOptionSingle<Pointf>(value) {}
|
2013-12-20 20:32:18 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coPoint; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionPoint(*this); }
|
|
|
|
ConfigOptionPoint& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionPoint &rhs) const { return this->value == rhs.value; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2013-12-20 19:54:11 +00:00
|
|
|
std::ostringstream ss;
|
2018-08-17 13:53:43 +00:00
|
|
|
ss << this->value(0);
|
2013-12-20 19:54:11 +00:00
|
|
|
ss << ",";
|
2018-08-17 13:53:43 +00:00
|
|
|
ss << this->value(1);
|
2013-12-20 19:54:11 +00:00
|
|
|
return ss.str();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-20 19:54:11 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-07-27 08:39:43 +00:00
|
|
|
UNUSED(append);
|
2018-05-17 08:30:20 +00:00
|
|
|
char dummy;
|
2018-08-17 13:53:43 +00:00
|
|
|
return sscanf(str.data(), " %lf , %lf %c", &this->value(0), &this->value(1), &dummy) == 2 ||
|
|
|
|
sscanf(str.data(), " %lf x %lf %c", &this->value(0), &this->value(1), &dummy) == 2;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-07 15:14:30 +00:00
|
|
|
};
|
|
|
|
|
2015-05-02 19:43:22 +00:00
|
|
|
class ConfigOptionPoints : public ConfigOptionVector<Pointf>
|
2013-12-21 13:27:58 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionPoints() : ConfigOptionVector<Pointf>() {}
|
|
|
|
explicit ConfigOptionPoints(size_t n, const Pointf &value) : ConfigOptionVector<Pointf>(n, value) {}
|
|
|
|
explicit ConfigOptionPoints(std::initializer_list<Pointf> il) : ConfigOptionVector<Pointf>(std::move(il)) {}
|
2018-04-05 14:10:44 +00:00
|
|
|
explicit ConfigOptionPoints(const std::vector<Pointf> &values) : ConfigOptionVector<Pointf>(values) {}
|
2017-11-01 18:30:05 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coPoints; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionPoints(*this); }
|
|
|
|
ConfigOptionPoints& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionPoints &rhs) const { return this->values == rhs.values; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2013-12-21 13:27:58 +00:00
|
|
|
std::ostringstream ss;
|
2014-01-04 23:36:33 +00:00
|
|
|
for (Pointfs::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
|
|
|
|
if (it - this->values.begin() != 0) ss << ",";
|
2018-08-17 13:53:43 +00:00
|
|
|
ss << (*it)(0);
|
2013-12-21 13:27:58 +00:00
|
|
|
ss << "x";
|
2018-08-17 13:53:43 +00:00
|
|
|
ss << (*it)(1);
|
2013-12-21 13:27:58 +00:00
|
|
|
}
|
|
|
|
return ss.str();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-21 13:27:58 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::vector<std::string> vserialize() const override
|
|
|
|
{
|
2015-05-02 19:43:22 +00:00
|
|
|
std::vector<std::string> vv;
|
|
|
|
for (Pointfs::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
|
|
|
|
std::ostringstream ss;
|
|
|
|
ss << *it;
|
|
|
|
vv.push_back(ss.str());
|
|
|
|
}
|
|
|
|
return vv;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2015-05-02 19:43:22 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-06-14 15:51:14 +00:00
|
|
|
if (! append)
|
|
|
|
this->values.clear();
|
2013-12-21 13:27:58 +00:00
|
|
|
std::istringstream is(str);
|
|
|
|
std::string point_str;
|
|
|
|
while (std::getline(is, point_str, ',')) {
|
2018-08-21 18:34:45 +00:00
|
|
|
Pointf point(Vec2d::Zero());
|
2014-08-04 09:34:53 +00:00
|
|
|
std::istringstream iss(point_str);
|
2015-01-28 15:08:50 +00:00
|
|
|
std::string coord_str;
|
|
|
|
if (std::getline(iss, coord_str, 'x')) {
|
2018-08-17 13:53:43 +00:00
|
|
|
std::istringstream(coord_str) >> point(0);
|
2015-01-28 15:08:50 +00:00
|
|
|
if (std::getline(iss, coord_str, 'x')) {
|
2018-08-17 13:53:43 +00:00
|
|
|
std::istringstream(coord_str) >> point(1);
|
2015-01-28 15:08:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
this->values.push_back(point);
|
2013-12-21 13:27:58 +00:00
|
|
|
}
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-21 13:27:58 +00:00
|
|
|
};
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionBool : public ConfigOptionSingle<bool>
|
2013-12-20 20:32:18 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionBool() : ConfigOptionSingle<bool>(false) {}
|
|
|
|
explicit ConfigOptionBool(bool _value) : ConfigOptionSingle<bool>(_value) {}
|
2013-12-20 20:32:18 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coBool; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
bool getBool() const override { return this->value; }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionBool(*this); }
|
|
|
|
ConfigOptionBool& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionBool &rhs) const { return this->value == rhs.value; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2013-12-20 20:32:18 +00:00
|
|
|
return std::string(this->value ? "1" : "0");
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-20 20:32:18 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-07-27 08:39:43 +00:00
|
|
|
UNUSED(append);
|
2013-12-20 20:32:18 +00:00
|
|
|
this->value = (str.compare("1") == 0);
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-20 20:32:18 +00:00
|
|
|
};
|
|
|
|
|
2017-09-19 11:55:48 +00:00
|
|
|
class ConfigOptionBools : public ConfigOptionVector<unsigned char>
|
2013-12-21 13:27:58 +00:00
|
|
|
{
|
2017-09-19 11:55:48 +00:00
|
|
|
public:
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionBools() : ConfigOptionVector<unsigned char>() {}
|
|
|
|
explicit ConfigOptionBools(size_t n, bool value) : ConfigOptionVector<unsigned char>(n, (unsigned char)value) {}
|
|
|
|
explicit ConfigOptionBools(std::initializer_list<bool> il) { values.reserve(il.size()); for (bool b : il) values.emplace_back((unsigned char)b); }
|
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coBools; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionBools(*this); }
|
|
|
|
ConfigOptionBools& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionBools &rhs) const { return this->values == rhs.values; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
2017-09-19 11:55:48 +00:00
|
|
|
bool& get_at(size_t i) {
|
|
|
|
assert(! this->values.empty());
|
|
|
|
return *reinterpret_cast<bool*>(&((i < this->values.size()) ? this->values[i] : this->values.front()));
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2017-09-19 11:55:48 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
//FIXME this smells, the parent class has the method declared returning (unsigned char&).
|
2017-09-19 11:55:48 +00:00
|
|
|
bool get_at(size_t i) const { return bool((i < this->values.size()) ? this->values[i] : this->values.front()); }
|
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string serialize() const override
|
|
|
|
{
|
2013-12-21 13:27:58 +00:00
|
|
|
std::ostringstream ss;
|
2017-09-19 11:55:48 +00:00
|
|
|
for (std::vector<unsigned char>::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
|
2013-12-21 13:27:58 +00:00
|
|
|
if (it - this->values.begin() != 0) ss << ",";
|
|
|
|
ss << (*it ? "1" : "0");
|
|
|
|
}
|
|
|
|
return ss.str();
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-21 13:27:58 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
std::vector<std::string> vserialize() const override
|
|
|
|
{
|
2015-05-02 19:43:22 +00:00
|
|
|
std::vector<std::string> vv;
|
2017-09-19 11:55:48 +00:00
|
|
|
for (std::vector<unsigned char>::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
|
2015-05-02 19:43:22 +00:00
|
|
|
std::ostringstream ss;
|
|
|
|
ss << (*it ? "1" : "0");
|
|
|
|
vv.push_back(ss.str());
|
|
|
|
}
|
|
|
|
return vv;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2015-05-02 19:43:22 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-06-14 15:51:14 +00:00
|
|
|
if (! append)
|
|
|
|
this->values.clear();
|
2013-12-21 13:27:58 +00:00
|
|
|
std::istringstream is(str);
|
|
|
|
std::string item_str;
|
|
|
|
while (std::getline(is, item_str, ',')) {
|
|
|
|
this->values.push_back(item_str.compare("1") == 0);
|
|
|
|
}
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-20 20:32:18 +00:00
|
|
|
};
|
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
// Map from an enum integer value to an enum name.
|
|
|
|
typedef std::vector<std::string> t_config_enum_names;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Map from an enum name to an enum integer value.
|
2013-12-21 15:15:41 +00:00
|
|
|
typedef std::map<std::string,int> t_config_enum_values;
|
|
|
|
|
2013-12-21 09:46:43 +00:00
|
|
|
template <class T>
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionEnum : public ConfigOptionSingle<T>
|
2013-12-20 20:32:18 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2015-12-07 18:39:49 +00:00
|
|
|
// by default, use the first value (0) of the T enum type
|
2017-11-01 18:30:05 +00:00
|
|
|
ConfigOptionEnum() : ConfigOptionSingle<T>(static_cast<T>(0)) {}
|
|
|
|
explicit ConfigOptionEnum(T _value) : ConfigOptionSingle<T>(_value) {}
|
2017-10-17 14:01:18 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coEnum; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionEnum<T>(*this); }
|
|
|
|
ConfigOptionEnum<T>& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionEnum<T> &rhs) const { return this->value == rhs.value; }
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
|
|
|
const t_config_enum_names& names = ConfigOptionEnum<T>::get_enum_names();
|
|
|
|
assert(static_cast<int>(this->value) < int(names.size()));
|
|
|
|
return names[static_cast<int>(this->value)];
|
|
|
|
}
|
2013-12-21 13:48:25 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-07-27 08:39:43 +00:00
|
|
|
UNUSED(append);
|
2017-10-17 14:01:18 +00:00
|
|
|
const t_config_enum_values &enum_keys_map = ConfigOptionEnum<T>::get_enum_values();
|
|
|
|
auto it = enum_keys_map.find(str);
|
|
|
|
if (it == enum_keys_map.end())
|
|
|
|
return false;
|
|
|
|
this->value = static_cast<T>(it->second);
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-21 13:48:25 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
static bool has(T value)
|
|
|
|
{
|
2017-10-17 15:48:04 +00:00
|
|
|
for (const std::pair<std::string, int> &kvp : ConfigOptionEnum<T>::get_enum_values())
|
2017-10-17 14:01:18 +00:00
|
|
|
if (kvp.second == value)
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Map from an enum name to an enum integer value.
|
|
|
|
static t_config_enum_names& get_enum_names()
|
|
|
|
{
|
|
|
|
static t_config_enum_names names;
|
|
|
|
if (names.empty()) {
|
|
|
|
// Initialize the map.
|
|
|
|
const t_config_enum_values &enum_keys_map = ConfigOptionEnum<T>::get_enum_values();
|
|
|
|
int cnt = 0;
|
2017-10-17 15:48:04 +00:00
|
|
|
for (const std::pair<std::string, int> &kvp : enum_keys_map)
|
2017-10-17 14:01:18 +00:00
|
|
|
cnt = std::max(cnt, kvp.second);
|
|
|
|
cnt += 1;
|
|
|
|
names.assign(cnt, "");
|
2017-10-17 15:48:04 +00:00
|
|
|
for (const std::pair<std::string, int> &kvp : enum_keys_map)
|
2017-10-17 14:01:18 +00:00
|
|
|
names[kvp.second] = kvp.first;
|
|
|
|
}
|
|
|
|
return names;
|
|
|
|
}
|
2016-09-13 11:30:00 +00:00
|
|
|
// Map from an enum name to an enum integer value.
|
2017-10-17 14:01:18 +00:00
|
|
|
static t_config_enum_values& get_enum_values();
|
2013-12-21 09:46:43 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Generic enum configuration value.
|
|
|
|
// We use this one in DynamicConfig objects when creating a config value object for ConfigOptionType == coEnum.
|
|
|
|
// In the StaticConfig, it is better to use the specialized ConfigOptionEnum<T> containers.
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionEnumGeneric : public ConfigOptionInt
|
2013-12-21 15:15:41 +00:00
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2017-10-17 14:01:18 +00:00
|
|
|
ConfigOptionEnumGeneric(const t_config_enum_values* keys_map = nullptr) : keys_map(keys_map) {}
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
const t_config_enum_values* keys_map;
|
2013-12-21 15:15:41 +00:00
|
|
|
|
2017-11-09 09:05:37 +00:00
|
|
|
static ConfigOptionType static_type() { return coEnum; }
|
|
|
|
ConfigOptionType type() const override { return static_type(); }
|
|
|
|
ConfigOption* clone() const override { return new ConfigOptionEnumGeneric(*this); }
|
2017-10-17 14:01:18 +00:00
|
|
|
ConfigOptionEnumGeneric& operator=(const ConfigOption *opt) { this->set(opt); return *this; }
|
|
|
|
bool operator==(const ConfigOptionEnumGeneric &rhs) const { return this->value == rhs.value; }
|
|
|
|
|
|
|
|
std::string serialize() const override
|
|
|
|
{
|
|
|
|
for (const auto &kvp : *this->keys_map)
|
|
|
|
if (kvp.second == this->value)
|
|
|
|
return kvp.first;
|
|
|
|
return std::string();
|
|
|
|
}
|
2013-12-20 20:32:18 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool deserialize(const std::string &str, bool append = false) override
|
|
|
|
{
|
2017-07-27 08:39:43 +00:00
|
|
|
UNUSED(append);
|
2017-10-17 14:01:18 +00:00
|
|
|
auto it = this->keys_map->find(str);
|
|
|
|
if (it == this->keys_map->end())
|
|
|
|
return false;
|
|
|
|
this->value = it->second;
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2017-10-17 14:01:18 +00:00
|
|
|
}
|
2013-12-07 15:14:30 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Definition of a configuration value for the purpose of GUI presentation, editing, value mapping and config file handling.
|
2013-12-07 15:14:30 +00:00
|
|
|
class ConfigOptionDef
|
|
|
|
{
|
2017-05-30 19:29:43 +00:00
|
|
|
public:
|
2016-09-13 11:30:00 +00:00
|
|
|
// What type? bool, int, string etc.
|
2017-10-17 14:01:18 +00:00
|
|
|
ConfigOptionType type = coNone;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Default value of this option. The default value object is owned by ConfigDef, it is released in its destructor.
|
2017-10-17 14:01:18 +00:00
|
|
|
ConfigOption *default_value = nullptr;
|
2016-09-13 11:30:00 +00:00
|
|
|
|
|
|
|
// Usually empty.
|
|
|
|
// Special values - "i_enum_open", "f_enum_open" to provide combo box for int or float selection,
|
|
|
|
// "select_open" - to open a selection dialog (currently only a serial port selection).
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string gui_type;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Usually empty. Otherwise "serialized" or "show_value"
|
|
|
|
// The flags may be combined.
|
|
|
|
// "serialized" - vector valued option is entered in a single edit field. Values are separated by a semicolon.
|
|
|
|
// "show_value" - even if enum_values / enum_labels are set, still display the value, not the enum label.
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string gui_flags;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Label of the GUI input field.
|
|
|
|
// In case the GUI input fields are grouped in some views, the label defines a short label of a grouped value,
|
|
|
|
// while full_label contains a label of a stand-alone field.
|
|
|
|
// The full label is shown, when adding an override parameter for an object or a modified object.
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string label;
|
|
|
|
std::string full_label;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Category of a configuration field, from the GUI perspective.
|
|
|
|
// One of: "Layers and Perimeters", "Infill", "Support material", "Speed", "Extruders", "Advanced", "Extrusion Width"
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string category;
|
2016-09-13 11:30:00 +00:00
|
|
|
// A tooltip text shown in the GUI.
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string tooltip;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Text right from the input field, usually a unit of measurement.
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string sidetext;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Format of this parameter on a command line.
|
2017-10-17 14:01:18 +00:00
|
|
|
std::string cli;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Set for type == coFloatOrPercent.
|
|
|
|
// It provides a link to a configuration value, of which this option provides a ratio.
|
|
|
|
// For example,
|
|
|
|
// For example external_perimeter_speed may be defined as a fraction of perimeter_speed.
|
2017-10-17 14:01:18 +00:00
|
|
|
t_config_option_key ratio_over;
|
2016-09-13 11:30:00 +00:00
|
|
|
// True for multiline strings.
|
2017-10-17 14:01:18 +00:00
|
|
|
bool multiline = false;
|
2016-09-13 11:30:00 +00:00
|
|
|
// For text input: If true, the GUI text box spans the complete page width.
|
2017-10-17 14:01:18 +00:00
|
|
|
bool full_width = false;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Not editable. Currently only used for the display of the number of threads.
|
2017-10-17 14:01:18 +00:00
|
|
|
bool readonly = false;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Height of a multiline GUI text box.
|
2017-10-17 14:01:18 +00:00
|
|
|
int height = -1;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Optional width of an input field.
|
2017-10-17 14:01:18 +00:00
|
|
|
int width = -1;
|
2016-09-13 11:30:00 +00:00
|
|
|
// <min, max> limit of a numeric input.
|
|
|
|
// If not set, the <min, max> is set to <INT_MIN, INT_MAX>
|
|
|
|
// By setting min=0, only nonnegative input is allowed.
|
2017-10-17 14:01:18 +00:00
|
|
|
int min = INT_MIN;
|
|
|
|
int max = INT_MAX;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Legacy names for this configuration option.
|
|
|
|
// Used when parsing legacy configuration file.
|
2017-10-17 14:01:18 +00:00
|
|
|
std::vector<t_config_option_key> aliases;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Sometimes a single value may well define multiple values in a "beginner" mode.
|
|
|
|
// Currently used for aliasing "solid_layers" to "top_solid_layers", "bottom_solid_layers".
|
2017-10-17 14:01:18 +00:00
|
|
|
std::vector<t_config_option_key> shortcut;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Definition of values / labels for a combo box.
|
|
|
|
// Mostly used for enums (when type == coEnum), but may be used for ints resp. floats, if gui_type is set to "i_enum_open" resp. "f_enum_open".
|
2017-10-17 14:01:18 +00:00
|
|
|
std::vector<std::string> enum_values;
|
|
|
|
std::vector<std::string> enum_labels;
|
2016-09-13 11:30:00 +00:00
|
|
|
// For enums (when type == coEnum). Maps enum_values to enums.
|
|
|
|
// Initialized by ConfigOptionEnum<xxx>::get_enum_values()
|
2017-10-17 14:01:18 +00:00
|
|
|
t_config_enum_values *enum_keys_map = nullptr;
|
2016-09-13 11:30:00 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
bool has_enum_value(const std::string &value) const {
|
|
|
|
for (const std::string &v : enum_values)
|
|
|
|
if (v == value)
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
2013-12-07 15:14:30 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Map from a config option name to its definition.
|
|
|
|
// The definition does not carry an actual value of the config option, only its constant default value.
|
|
|
|
// t_config_option_key is std::string
|
2013-12-07 15:14:30 +00:00
|
|
|
typedef std::map<t_config_option_key,ConfigOptionDef> t_optiondef_map;
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Definition of configuration values for the purpose of GUI presentation, editing, value mapping and config file handling.
|
|
|
|
// The configuration definition is static: It does not carry the actual configuration values,
|
|
|
|
// but it carries the defaults of the configuration values.
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigDef
|
|
|
|
{
|
2017-05-30 15:04:36 +00:00
|
|
|
public:
|
2015-12-07 18:39:49 +00:00
|
|
|
t_optiondef_map options;
|
2017-05-30 15:04:36 +00:00
|
|
|
~ConfigDef() { for (auto &opt : this->options) delete opt.second.default_value; }
|
2017-10-17 14:01:18 +00:00
|
|
|
ConfigOptionDef* add(const t_config_option_key &opt_key, ConfigOptionType type) {
|
2017-05-30 15:04:36 +00:00
|
|
|
ConfigOptionDef* opt = &this->options[opt_key];
|
|
|
|
opt->type = type;
|
|
|
|
return opt;
|
|
|
|
}
|
2017-10-17 14:01:18 +00:00
|
|
|
bool has(const t_config_option_key &opt_key) const { return this->options.count(opt_key) > 0; }
|
|
|
|
const ConfigOptionDef* get(const t_config_option_key &opt_key) const {
|
2017-05-30 15:04:36 +00:00
|
|
|
t_optiondef_map::iterator it = const_cast<ConfigDef*>(this)->options.find(opt_key);
|
|
|
|
return (it == this->options.end()) ? nullptr : &it->second;
|
|
|
|
}
|
2015-12-07 18:39:49 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// An abstract configuration store.
|
2013-12-07 15:14:30 +00:00
|
|
|
class ConfigBase
|
|
|
|
{
|
2017-05-30 15:04:36 +00:00
|
|
|
public:
|
2016-09-13 11:30:00 +00:00
|
|
|
// Definition of configuration values for the purpose of GUI presentation, editing, value mapping and config file handling.
|
|
|
|
// The configuration definition is static: It does not carry the actual configuration values,
|
|
|
|
// but it carries the defaults of the configuration values.
|
2013-12-21 15:15:41 +00:00
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
ConfigBase() {}
|
|
|
|
virtual ~ConfigBase() {}
|
|
|
|
|
|
|
|
// Virtual overridables:
|
|
|
|
public:
|
|
|
|
// Static configuration definition. Any value stored into this ConfigBase shall have its definition here.
|
|
|
|
virtual const ConfigDef* def() const = 0;
|
|
|
|
// Find ando/or create a ConfigOption instance for a given name.
|
|
|
|
virtual ConfigOption* optptr(const t_config_option_key &opt_key, bool create = false) = 0;
|
|
|
|
// Collect names of all configuration values maintained by this configuration store.
|
|
|
|
virtual t_config_option_keys keys() const = 0;
|
|
|
|
protected:
|
|
|
|
// Verify whether the opt_key has not been obsoleted or renamed.
|
|
|
|
// Both opt_key and value may be modified by handle_legacy().
|
|
|
|
// If the opt_key is no more valid in this version of Slic3r, opt_key is cleared by handle_legacy().
|
|
|
|
// handle_legacy() is called internally by set_deserialize().
|
|
|
|
virtual void handle_legacy(t_config_option_key &opt_key, std::string &value) const {}
|
|
|
|
|
|
|
|
public:
|
|
|
|
// Non-virtual methods:
|
2017-05-30 15:04:36 +00:00
|
|
|
bool has(const t_config_option_key &opt_key) const { return this->option(opt_key) != nullptr; }
|
|
|
|
const ConfigOption* option(const t_config_option_key &opt_key) const
|
|
|
|
{ return const_cast<ConfigBase*>(this)->option(opt_key, false); }
|
|
|
|
ConfigOption* option(const t_config_option_key &opt_key, bool create = false)
|
|
|
|
{ return this->optptr(opt_key, create); }
|
2017-11-09 09:05:37 +00:00
|
|
|
template<typename TYPE>
|
|
|
|
TYPE* option(const t_config_option_key &opt_key, bool create = false)
|
|
|
|
{
|
|
|
|
ConfigOption *opt = this->optptr(opt_key, create);
|
2017-11-10 16:42:30 +00:00
|
|
|
assert(opt == nullptr || opt->type() == TYPE::static_type());
|
|
|
|
return (opt == nullptr || opt->type() != TYPE::static_type()) ? nullptr : static_cast<TYPE*>(opt);
|
2017-11-09 09:05:37 +00:00
|
|
|
}
|
|
|
|
template<typename TYPE>
|
|
|
|
const TYPE* option(const t_config_option_key &opt_key) const
|
|
|
|
{ return const_cast<ConfigBase*>(this)->option<TYPE>(opt_key, false); }
|
2017-10-17 14:01:18 +00:00
|
|
|
// Apply all keys of other ConfigBase defined by this->def() to this ConfigBase.
|
|
|
|
// An UnknownOptionException is thrown in case some option keys of other are not defined by this->def(),
|
|
|
|
// or this ConfigBase is of a StaticConfig type and it does not support some of the keys, and ignore_nonexistent is not set.
|
|
|
|
void apply(const ConfigBase &other, bool ignore_nonexistent = false) { this->apply_only(other, other.keys(), ignore_nonexistent); }
|
|
|
|
// Apply explicitely enumerated keys of other ConfigBase defined by this->def() to this ConfigBase.
|
|
|
|
// An UnknownOptionException is thrown in case some option keys are not defined by this->def(),
|
|
|
|
// or this ConfigBase is of a StaticConfig type and it does not support some of the keys, and ignore_nonexistent is not set.
|
|
|
|
void apply_only(const ConfigBase &other, const t_config_option_keys &keys, bool ignore_nonexistent = false);
|
2017-05-30 15:04:36 +00:00
|
|
|
bool equals(const ConfigBase &other) const { return this->diff(other).empty(); }
|
|
|
|
t_config_option_keys diff(const ConfigBase &other) const;
|
2018-04-18 11:32:21 +00:00
|
|
|
// Use deep_diff to correct return of changed options,
|
|
|
|
// considering individual options for each extruder
|
|
|
|
t_config_option_keys deep_diff(const ConfigBase &other) const;
|
2018-03-16 11:56:03 +00:00
|
|
|
t_config_option_keys equal(const ConfigBase &other) const;
|
2015-12-02 18:32:57 +00:00
|
|
|
std::string serialize(const t_config_option_key &opt_key) const;
|
2017-10-17 14:01:18 +00:00
|
|
|
// Set a configuration value from a string, it will call an overridable handle_legacy()
|
|
|
|
// to resolve renamed and removed configuration keys.
|
|
|
|
bool set_deserialize(const t_config_option_key &opt_key, const std::string &str, bool append = false);
|
2016-09-13 11:30:00 +00:00
|
|
|
|
2016-11-01 12:41:24 +00:00
|
|
|
double get_abs_value(const t_config_option_key &opt_key) const;
|
|
|
|
double get_abs_value(const t_config_option_key &opt_key, double ratio_over) const;
|
2015-07-01 16:18:25 +00:00
|
|
|
void setenv_();
|
2017-06-14 15:51:14 +00:00
|
|
|
void load(const std::string &file);
|
2017-10-27 14:11:06 +00:00
|
|
|
void load_from_ini(const std::string &file);
|
2018-02-13 14:19:55 +00:00
|
|
|
void load_from_gcode_file(const std::string &file);
|
|
|
|
void load_from_gcode_string(const char* str);
|
2017-11-01 18:30:05 +00:00
|
|
|
void load(const boost::property_tree::ptree &tree);
|
2017-06-14 15:51:14 +00:00
|
|
|
void save(const std::string &file) const;
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
// Set a configuration value from a string.
|
|
|
|
bool set_deserialize_raw(const t_config_option_key &opt_key_src, const std::string &str, bool append);
|
2013-12-07 15:14:30 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Configuration store with dynamic number of configuration values.
|
|
|
|
// In Slic3r, the dynamic config is mostly used at the user interface layer.
|
2015-12-07 18:39:49 +00:00
|
|
|
class DynamicConfig : public virtual ConfigBase
|
2013-12-07 15:14:30 +00:00
|
|
|
{
|
2017-05-30 15:04:36 +00:00
|
|
|
public:
|
2017-11-17 10:15:46 +00:00
|
|
|
DynamicConfig() {}
|
2017-10-17 14:01:18 +00:00
|
|
|
DynamicConfig(const DynamicConfig& other) { *this = other; }
|
|
|
|
DynamicConfig(DynamicConfig&& other) : options(std::move(other.options)) { other.options.clear(); }
|
|
|
|
virtual ~DynamicConfig() { clear(); }
|
|
|
|
|
2017-11-17 10:15:46 +00:00
|
|
|
// Copy a content of one DynamicConfig to another DynamicConfig.
|
|
|
|
// If rhs.def() is not null, then it has to be equal to this->def().
|
|
|
|
DynamicConfig& operator=(const DynamicConfig &rhs)
|
2017-10-17 14:01:18 +00:00
|
|
|
{
|
2017-11-17 10:15:46 +00:00
|
|
|
assert(this->def() == nullptr || this->def() == rhs.def());
|
2017-10-17 14:01:18 +00:00
|
|
|
this->clear();
|
2017-11-17 10:15:46 +00:00
|
|
|
for (const auto &kvp : rhs.options)
|
2017-10-17 14:01:18 +00:00
|
|
|
this->options[kvp.first] = kvp.second->clone();
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2017-11-17 10:15:46 +00:00
|
|
|
// Move a content of one DynamicConfig to another DynamicConfig.
|
|
|
|
// If rhs.def() is not null, then it has to be equal to this->def().
|
|
|
|
DynamicConfig& operator=(DynamicConfig &&rhs)
|
2017-10-17 14:01:18 +00:00
|
|
|
{
|
2017-11-17 10:15:46 +00:00
|
|
|
assert(this->def() == nullptr || this->def() == rhs.def());
|
2017-10-17 14:01:18 +00:00
|
|
|
this->clear();
|
2017-11-17 10:15:46 +00:00
|
|
|
this->options = std::move(rhs.options);
|
|
|
|
rhs.options.clear();
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add a content of one DynamicConfig to another DynamicConfig.
|
|
|
|
// If rhs.def() is not null, then it has to be equal to this->def().
|
|
|
|
DynamicConfig& operator+=(const DynamicConfig &rhs)
|
|
|
|
{
|
|
|
|
assert(this->def() == nullptr || this->def() == rhs.def());
|
|
|
|
for (const auto &kvp : rhs.options) {
|
|
|
|
auto it = this->options.find(kvp.first);
|
|
|
|
if (it == this->options.end())
|
|
|
|
this->options[kvp.first] = kvp.second->clone();
|
|
|
|
else {
|
|
|
|
assert(it->second->type() == kvp.second->type());
|
|
|
|
if (it->second->type() == kvp.second->type())
|
|
|
|
*it->second = *kvp.second;
|
|
|
|
else {
|
|
|
|
delete it->second;
|
|
|
|
it->second = kvp.second->clone();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Move a content of one DynamicConfig to another DynamicConfig.
|
|
|
|
// If rhs.def() is not null, then it has to be equal to this->def().
|
|
|
|
DynamicConfig& operator+=(DynamicConfig &&rhs)
|
|
|
|
{
|
|
|
|
assert(this->def() == nullptr || this->def() == rhs.def());
|
|
|
|
for (const auto &kvp : rhs.options) {
|
|
|
|
auto it = this->options.find(kvp.first);
|
|
|
|
if (it == this->options.end()) {
|
|
|
|
this->options[kvp.first] = kvp.second;
|
|
|
|
} else {
|
|
|
|
assert(it->second->type() == kvp.second->type());
|
|
|
|
delete it->second;
|
|
|
|
it->second = kvp.second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rhs.options.clear();
|
2017-10-17 14:01:18 +00:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2017-11-01 18:30:05 +00:00
|
|
|
bool operator==(const DynamicConfig &rhs) const;
|
|
|
|
bool operator!=(const DynamicConfig &rhs) const { return ! (*this == rhs); }
|
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
void swap(DynamicConfig &other)
|
|
|
|
{
|
|
|
|
std::swap(this->options, other.options);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clear()
|
|
|
|
{
|
|
|
|
for (auto &opt : this->options)
|
|
|
|
delete opt.second;
|
|
|
|
this->options.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool erase(const t_config_option_key &opt_key)
|
|
|
|
{
|
|
|
|
auto it = this->options.find(opt_key);
|
|
|
|
if (it == this->options.end())
|
|
|
|
return false;
|
|
|
|
delete it->second;
|
|
|
|
this->options.erase(it);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-11-17 10:15:46 +00:00
|
|
|
// Allow DynamicConfig to be instantiated on ints own without a definition.
|
|
|
|
// If the definition is not defined, the method requiring the definition will throw NoDefinitionException.
|
|
|
|
const ConfigDef* def() const override { return nullptr; };
|
|
|
|
template<class T> T* opt(const t_config_option_key &opt_key, bool create = false)
|
2017-10-17 14:01:18 +00:00
|
|
|
{ return dynamic_cast<T*>(this->option(opt_key, create)); }
|
2018-01-04 14:38:06 +00:00
|
|
|
template<class T> const T* opt(const t_config_option_key &opt_key) const
|
|
|
|
{ return dynamic_cast<const T*>(this->option(opt_key)); }
|
2017-10-17 14:01:18 +00:00
|
|
|
// Overrides ConfigBase::optptr(). Find ando/or create a ConfigOption instance for a given name.
|
|
|
|
ConfigOption* optptr(const t_config_option_key &opt_key, bool create = false) override;
|
|
|
|
// Overrides ConfigBase::keys(). Collect names of all configuration values maintained by this configuration store.
|
|
|
|
t_config_option_keys keys() const override;
|
2017-05-30 15:04:36 +00:00
|
|
|
|
2017-11-17 10:15:46 +00:00
|
|
|
// Set a value for an opt_key. Returns true if the value did not exist yet.
|
|
|
|
// This DynamicConfig will take ownership of opt.
|
|
|
|
// Be careful, as this method does not test the existence of opt_key in this->def().
|
|
|
|
bool set_key_value(const std::string &opt_key, ConfigOption *opt)
|
|
|
|
{
|
|
|
|
auto it = this->options.find(opt_key);
|
|
|
|
if (it == this->options.end()) {
|
|
|
|
this->options[opt_key] = opt;
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
delete it->second;
|
|
|
|
it->second = opt;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-11-09 14:10:20 +00:00
|
|
|
std::string& opt_string(const t_config_option_key &opt_key, bool create = false) { return this->option<ConfigOptionString>(opt_key, create)->value; }
|
2017-09-19 11:55:48 +00:00
|
|
|
const std::string& opt_string(const t_config_option_key &opt_key) const { return const_cast<DynamicConfig*>(this)->opt_string(opt_key); }
|
2017-11-09 14:10:20 +00:00
|
|
|
std::string& opt_string(const t_config_option_key &opt_key, unsigned int idx) { return this->option<ConfigOptionStrings>(opt_key)->get_at(idx); }
|
2017-09-19 11:55:48 +00:00
|
|
|
const std::string& opt_string(const t_config_option_key &opt_key, unsigned int idx) const { return const_cast<DynamicConfig*>(this)->opt_string(opt_key, idx); }
|
|
|
|
|
2017-11-09 14:10:20 +00:00
|
|
|
double& opt_float(const t_config_option_key &opt_key) { return this->option<ConfigOptionFloat>(opt_key)->value; }
|
2017-10-17 14:01:18 +00:00
|
|
|
const double opt_float(const t_config_option_key &opt_key) const { return dynamic_cast<const ConfigOptionFloat*>(this->option(opt_key))->value; }
|
2017-11-09 14:10:20 +00:00
|
|
|
double& opt_float(const t_config_option_key &opt_key, unsigned int idx) { return this->option<ConfigOptionFloats>(opt_key)->get_at(idx); }
|
2017-10-17 14:01:18 +00:00
|
|
|
const double opt_float(const t_config_option_key &opt_key, unsigned int idx) const { return dynamic_cast<const ConfigOptionFloats*>(this->option(opt_key))->get_at(idx); }
|
|
|
|
|
2017-11-09 14:10:20 +00:00
|
|
|
int& opt_int(const t_config_option_key &opt_key) { return this->option<ConfigOptionInt>(opt_key)->value; }
|
2017-10-17 14:01:18 +00:00
|
|
|
const int opt_int(const t_config_option_key &opt_key) const { return dynamic_cast<const ConfigOptionInt*>(this->option(opt_key))->value; }
|
2017-11-09 14:10:20 +00:00
|
|
|
int& opt_int(const t_config_option_key &opt_key, unsigned int idx) { return this->option<ConfigOptionInts>(opt_key)->get_at(idx); }
|
2017-10-17 14:01:18 +00:00
|
|
|
const int opt_int(const t_config_option_key &opt_key, unsigned int idx) const { return dynamic_cast<const ConfigOptionInts*>(this->option(opt_key))->get_at(idx); }
|
|
|
|
|
2017-11-09 14:10:20 +00:00
|
|
|
bool opt_bool(const t_config_option_key &opt_key) const { return this->option<ConfigOptionBool>(opt_key)->value != 0; }
|
|
|
|
bool opt_bool(const t_config_option_key &opt_key, unsigned int idx) const { return this->option<ConfigOptionBools>(opt_key)->get_at(idx) != 0; }
|
2017-11-09 09:05:37 +00:00
|
|
|
|
2017-05-30 15:04:36 +00:00
|
|
|
private:
|
2013-12-20 15:37:28 +00:00
|
|
|
typedef std::map<t_config_option_key,ConfigOption*> t_options_map;
|
|
|
|
t_options_map options;
|
2013-12-07 15:14:30 +00:00
|
|
|
};
|
|
|
|
|
2017-10-17 14:01:18 +00:00
|
|
|
/// Configuration store with a static definition of configuration values.
|
|
|
|
/// In Slic3r, the static configuration stores are during the slicing / g-code generation for efficiency reasons,
|
|
|
|
/// because the configuration values could be accessed directly.
|
2015-12-07 18:39:49 +00:00
|
|
|
class StaticConfig : public virtual ConfigBase
|
2013-12-07 15:14:30 +00:00
|
|
|
{
|
2017-05-30 15:04:36 +00:00
|
|
|
public:
|
2017-10-17 14:01:18 +00:00
|
|
|
StaticConfig() {}
|
|
|
|
/// Gets list of config option names for each config option of this->def, which has a static counter-part defined by the derived object
|
|
|
|
/// and which could be resolved by this->optptr(key) call.
|
2015-07-01 16:18:25 +00:00
|
|
|
t_config_option_keys keys() const;
|
2017-10-17 14:01:18 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
/// Set all statically defined config options to their defaults defined by this->def().
|
2015-12-07 18:39:49 +00:00
|
|
|
void set_defaults();
|
2013-12-07 15:14:30 +00:00
|
|
|
};
|
|
|
|
|
2017-06-14 15:51:14 +00:00
|
|
|
/// Specialization of std::exception to indicate that an unknown config option has been encountered.
|
2018-03-09 15:37:33 +00:00
|
|
|
class UnknownOptionException : public std::runtime_error {
|
2017-10-17 14:01:18 +00:00
|
|
|
public:
|
2018-03-09 15:37:33 +00:00
|
|
|
UnknownOptionException() :
|
|
|
|
std::runtime_error("Unknown option exception") {}
|
|
|
|
UnknownOptionException(const std::string &opt_key) :
|
|
|
|
std::runtime_error(std::string("Unknown option exception: ") + opt_key) {}
|
2017-10-17 14:01:18 +00:00
|
|
|
};
|
2017-06-14 15:51:14 +00:00
|
|
|
|
2017-11-17 10:15:46 +00:00
|
|
|
/// Indicate that the ConfigBase derived class does not provide config definition (the method def() returns null).
|
2018-03-09 15:37:33 +00:00
|
|
|
class NoDefinitionException : public std::runtime_error
|
2017-11-17 10:15:46 +00:00
|
|
|
{
|
|
|
|
public:
|
2018-03-09 15:37:33 +00:00
|
|
|
NoDefinitionException() :
|
|
|
|
std::runtime_error("No definition exception") {}
|
|
|
|
NoDefinitionException(const std::string &opt_key) :
|
|
|
|
std::runtime_error(std::string("No definition exception: ") + opt_key) {}
|
2017-11-17 10:15:46 +00:00
|
|
|
};
|
|
|
|
|
2013-12-07 15:14:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|