PrusaSlicer-NonPlainar/xs/src/libslic3r/Config.hpp

596 lines
17 KiB
C++
Raw Normal View History

#ifndef slic3r_Config_hpp_
#define slic3r_Config_hpp_
#include <map>
#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>
#include <string>
#include <vector>
#include "libslic3r.h"
2013-12-20 19:54:11 +00:00
#include "Point.hpp"
2013-12-20 15:37:28 +00:00
namespace Slic3r {
typedef std::string t_config_option_key;
typedef std::vector<std::string> t_config_option_keys;
2013-12-20 15:37:28 +00:00
class ConfigOption {
public:
virtual ~ConfigOption() {};
2014-01-05 12:16:13 +00:00
virtual std::string serialize() const = 0;
virtual bool deserialize(std::string str) = 0;
virtual void set(const ConfigOption &option) = 0;
virtual int getInt() const { return 0; };
virtual double getFloat() const { return 0; };
virtual bool getBool() const { return false; };
virtual void setInt(int val) {};
2015-12-02 18:32:57 +00:00
friend bool operator== (const ConfigOption &a, const ConfigOption &b);
friend bool operator!= (const ConfigOption &a, const ConfigOption &b);
2013-12-20 15:37:28 +00:00
};
template <class T>
class ConfigOptionSingle : public ConfigOption {
public:
T value;
ConfigOptionSingle(T _value) : value(_value) {};
operator T() const { return this->value; };
void set(const ConfigOption &option) {
const ConfigOptionSingle<T>* other = dynamic_cast< const ConfigOptionSingle<T>* >(&option);
if (other != NULL) this->value = other->value;
};
};
2015-05-02 19:43:22 +00:00
class ConfigOptionVectorBase : public ConfigOption {
public:
virtual ~ConfigOptionVectorBase() {};
virtual std::vector<std::string> vserialize() const = 0;
};
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
{
public:
virtual ~ConfigOptionVector() {};
std::vector<T> values;
void set(const ConfigOption &option) {
2015-12-08 09:53:57 +00:00
const ConfigOptionVector<T>* other = dynamic_cast< const ConfigOptionVector<T>* >(&option);
if (other != NULL) this->values = other->values;
};
2014-04-27 21:19:03 +00:00
T get_at(size_t i) const {
2014-01-04 23:36:33 +00:00
try {
return this->values.at(i);
} catch (const std::out_of_range& oor) {
return this->values.front();
}
};
};
class ConfigOptionFloat : public ConfigOptionSingle<double>
{
public:
2015-12-18 12:40:57 +00:00
ConfigOptionFloat() : ConfigOptionSingle<double>(0) {};
ConfigOptionFloat(double _value) : ConfigOptionSingle<double>(_value) {};
2013-12-20 19:54:11 +00:00
double getFloat() const { return this->value; };
2013-12-20 19:54:11 +00:00
2014-01-05 12:16:13 +00:00
std::string serialize() const {
2013-12-20 19:54:11 +00:00
std::ostringstream ss;
ss << this->value;
return ss.str();
};
bool deserialize(std::string str) {
std::istringstream iss(str);
return iss >> this->value;
2013-12-20 19:54:11 +00:00
};
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
{
public:
2014-01-05 12:16:13 +00:00
std::string serialize() const {
2013-12-21 13:27:58 +00:00
std::ostringstream ss;
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();
};
2015-05-02 19:43:22 +00:00
std::vector<std::string> vserialize() const {
std::vector<std::string> vv;
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;
};
bool deserialize(std::string str) {
2013-12-21 13:27:58 +00:00
this->values.clear();
std::istringstream is(str);
std::string item_str;
while (std::getline(is, item_str, ',')) {
std::istringstream iss(item_str);
double value;
iss >> value;
this->values.push_back(value);
2013-12-21 13:27:58 +00:00
}
return true;
2013-12-21 13:27:58 +00:00
};
};
class ConfigOptionInt : public ConfigOptionSingle<int>
2013-12-20 15:37:28 +00:00
{
public:
2015-12-18 12:40:57 +00:00
ConfigOptionInt() : ConfigOptionSingle<int>(0) {};
ConfigOptionInt(double _value) : ConfigOptionSingle<int>(_value) {};
2013-12-20 19:54:11 +00:00
int getInt() const { return this->value; };
void setInt(int val) { this->value = val; };
2013-12-20 19:54:11 +00:00
2014-01-05 12:16:13 +00:00
std::string serialize() const {
2013-12-20 19:54:11 +00:00
std::ostringstream ss;
ss << this->value;
return ss.str();
};
bool deserialize(std::string str) {
std::istringstream iss(str);
return iss >> this->value;
2013-12-20 19:54:11 +00:00
};
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
{
public:
2014-01-05 12:16:13 +00:00
std::string serialize() const {
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();
};
2015-05-02 19:43:22 +00:00
std::vector<std::string> vserialize() const {
std::vector<std::string> vv;
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;
};
bool deserialize(std::string str) {
2013-12-21 13:27:58 +00:00
this->values.clear();
std::istringstream is(str);
std::string item_str;
while (std::getline(is, item_str, ',')) {
std::istringstream iss(item_str);
int value;
iss >> value;
this->values.push_back(value);
2013-12-21 13:27:58 +00:00
}
return true;
2013-12-21 13:27:58 +00:00
};
};
class ConfigOptionString : public ConfigOptionSingle<std::string>
2013-12-20 15:37:28 +00:00
{
public:
2015-12-18 12:40:57 +00:00
ConfigOptionString() : ConfigOptionSingle<std::string>("") {};
ConfigOptionString(std::string _value) : ConfigOptionSingle<std::string>(_value) {};
2013-12-20 19:54:11 +00:00
2014-01-05 12:16:13 +00:00
std::string serialize() const {
2013-12-20 19:54:11 +00:00
std::string str = this->value;
// s/\R/\\n/g
size_t pos = 0;
while ((pos = str.find("\n", pos)) != std::string::npos || (pos = str.find("\r", pos)) != std::string::npos) {
str.replace(pos, 1, "\\n");
pos += 2; // length of "\\n"
}
return str;
};
bool deserialize(std::string str) {
2013-12-20 19:54:11 +00:00
// s/\\n/\n/g
size_t pos = 0;
while ((pos = str.find("\\n", pos)) != std::string::npos) {
str.replace(pos, 2, "\n");
pos += 1; // length of "\n"
}
this->value = str;
return true;
2013-12-20 19:54:11 +00:00
};
2013-12-20 15:37:28 +00:00
};
// semicolon-separated strings
2015-05-02 19:43:22 +00:00
class ConfigOptionStrings : public ConfigOptionVector<std::string>
{
public:
2014-01-05 12:16:13 +00:00
std::string serialize() const {
std::ostringstream ss;
for (std::vector<std::string>::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
if (it - this->values.begin() != 0) ss << ";";
ss << *it;
}
return ss.str();
};
2015-05-02 19:43:22 +00:00
std::vector<std::string> vserialize() const {
return this->values;
};
bool deserialize(std::string str) {
this->values.clear();
std::istringstream is(str);
std::string item_str;
while (std::getline(is, item_str, ';')) {
this->values.push_back(item_str);
}
return true;
};
};
class ConfigOptionPercent : public ConfigOptionFloat
2014-03-22 15:23:33 +00:00
{
public:
ConfigOptionPercent() : ConfigOptionFloat(0) {};
ConfigOptionPercent(double _value) : ConfigOptionFloat(_value) {};
2014-03-22 15:23:33 +00:00
double get_abs_value(double ratio_over) const {
return ratio_over * this->value / 100;
};
std::string serialize() const {
std::ostringstream ss;
ss << this->value;
std::string s(ss.str());
s += "%";
return s;
};
bool deserialize(std::string str) {
2014-03-22 15:23:33 +00:00
// don't try to parse the trailing % since it's optional
std::istringstream iss(str);
return iss >> this->value;
2014-03-22 15:23:33 +00:00
};
};
class ConfigOptionFloatOrPercent : public ConfigOptionPercent
2013-12-20 15:37:28 +00:00
{
public:
bool percent;
ConfigOptionFloatOrPercent() : ConfigOptionPercent(0), percent(false) {};
ConfigOptionFloatOrPercent(double _value, bool _percent)
: ConfigOptionPercent(_value), percent(_percent) {};
void set(const ConfigOption &option) {
const ConfigOptionFloatOrPercent* other = dynamic_cast< const ConfigOptionFloatOrPercent* >(&option);
if (other != NULL) {
this->value = other->value;
this->percent = other->percent;
}
};
2013-12-20 19:54:11 +00:00
2014-01-05 12:16:13 +00:00
double get_abs_value(double ratio_over) const {
if (this->percent) {
return ratio_over * this->value / 100;
} else {
return this->value;
}
};
std::string serialize() const {
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;
};
bool deserialize(std::string str) {
this->percent = str.find_first_of("%") != std::string::npos;
std::istringstream iss(str);
return iss >> this->value;
2013-12-20 19:54:11 +00:00
};
};
class ConfigOptionPoint : public ConfigOptionSingle<Pointf>
2013-12-20 19:54:11 +00:00
{
public:
2015-12-18 12:40:57 +00:00
ConfigOptionPoint() : ConfigOptionSingle<Pointf>(Pointf(0,0)) {};
ConfigOptionPoint(Pointf _value) : ConfigOptionSingle<Pointf>(_value) {};
2013-12-20 20:32:18 +00:00
2014-01-05 12:16:13 +00:00
std::string serialize() const {
2013-12-20 19:54:11 +00:00
std::ostringstream ss;
ss << this->value.x;
2013-12-20 19:54:11 +00:00
ss << ",";
ss << this->value.y;
2013-12-20 19:54:11 +00:00
return ss.str();
};
bool deserialize(std::string str) {
std::istringstream iss(str);
iss >> this->value.x;
iss.ignore(std::numeric_limits<std::streamsize>::max(), ',');
iss.ignore(std::numeric_limits<std::streamsize>::max(), 'x');
iss >> this->value.y;
return true;
2013-12-20 19:54:11 +00:00
};
};
2015-05-02 19:43:22 +00:00
class ConfigOptionPoints : public ConfigOptionVector<Pointf>
2013-12-21 13:27:58 +00:00
{
public:
2014-01-05 12:16:13 +00:00
std::string serialize() const {
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 << ",";
2013-12-21 13:27:58 +00:00
ss << it->x;
ss << "x";
ss << it->y;
}
return ss.str();
};
2015-05-02 19:43:22 +00:00
std::vector<std::string> vserialize() const {
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;
};
bool deserialize(std::string str) {
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, ',')) {
Pointf point;
std::istringstream iss(point_str);
std::string coord_str;
if (std::getline(iss, coord_str, 'x')) {
std::istringstream(coord_str) >> point.x;
if (std::getline(iss, coord_str, 'x')) {
std::istringstream(coord_str) >> point.y;
}
}
this->values.push_back(point);
2013-12-21 13:27:58 +00:00
}
return true;
2013-12-21 13:27:58 +00:00
};
};
class ConfigOptionBool : public ConfigOptionSingle<bool>
2013-12-20 20:32:18 +00:00
{
public:
2015-12-18 12:40:57 +00:00
ConfigOptionBool() : ConfigOptionSingle<bool>(false) {};
ConfigOptionBool(bool _value) : ConfigOptionSingle<bool>(_value) {};
2013-12-20 20:32:18 +00:00
bool getBool() const { return this->value; };
2013-12-20 20:32:18 +00:00
2014-01-05 12:16:13 +00:00
std::string serialize() const {
2013-12-20 20:32:18 +00:00
return std::string(this->value ? "1" : "0");
};
bool deserialize(std::string str) {
2013-12-20 20:32:18 +00:00
this->value = (str.compare("1") == 0);
return true;
2013-12-20 20:32:18 +00:00
};
};
2015-05-02 19:43:22 +00:00
class ConfigOptionBools : public ConfigOptionVector<bool>
2013-12-21 13:27:58 +00:00
{
public:
2014-01-05 12:16:13 +00:00
std::string serialize() const {
2013-12-21 13:27:58 +00:00
std::ostringstream ss;
for (std::vector<bool>::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
if (it - this->values.begin() != 0) ss << ",";
ss << (*it ? "1" : "0");
}
return ss.str();
};
2015-05-02 19:43:22 +00:00
std::vector<std::string> vserialize() const {
std::vector<std::string> vv;
for (std::vector<bool>::const_iterator it = this->values.begin(); it != this->values.end(); ++it) {
std::ostringstream ss;
ss << (*it ? "1" : "0");
vv.push_back(ss.str());
}
return vv;
};
bool deserialize(std::string str) {
2013-12-21 13:27:58 +00:00
this->values.clear();
std::istringstream is(str);
std::string item_str;
while (std::getline(is, item_str, ',')) {
this->values.push_back(item_str.compare("1") == 0);
}
return true;
2013-12-21 13:27:58 +00:00
};
2013-12-20 20:32:18 +00:00
};
typedef std::map<std::string,int> t_config_enum_values;
2013-12-21 09:46:43 +00:00
template <class T>
class ConfigOptionEnum : public ConfigOptionSingle<T>
2013-12-20 20:32:18 +00:00
{
public:
// by default, use the first value (0) of the T enum type
ConfigOptionEnum() : ConfigOptionSingle<T>(static_cast<T>(0)) {};
ConfigOptionEnum(T _value) : ConfigOptionSingle<T>(_value) {};
2013-12-20 20:32:18 +00:00
2014-01-05 12:16:13 +00:00
std::string serialize() const {
t_config_enum_values enum_keys_map = ConfigOptionEnum<T>::get_enum_values();
for (t_config_enum_values::iterator it = enum_keys_map.begin(); it != enum_keys_map.end(); ++it) {
if (it->second == static_cast<int>(this->value)) return it->first;
}
return "";
};
bool deserialize(std::string str) {
t_config_enum_values enum_keys_map = ConfigOptionEnum<T>::get_enum_values();
if (enum_keys_map.count(str) == 0) return false;
this->value = static_cast<T>(enum_keys_map[str]);
return true;
};
static t_config_enum_values get_enum_values();
2013-12-21 09:46:43 +00:00
};
/* We use this one in DynamicConfig objects, otherwise it's better to use
the specialized ConfigOptionEnum<T> containers. */
class ConfigOptionEnumGeneric : public ConfigOptionInt
{
public:
const t_config_enum_values* keys_map;
2014-01-05 12:16:13 +00:00
std::string serialize() const {
for (t_config_enum_values::const_iterator it = this->keys_map->begin(); it != this->keys_map->end(); ++it) {
if (it->second == this->value) return it->first;
}
return "";
};
2013-12-20 20:32:18 +00:00
bool deserialize(std::string str) {
if (this->keys_map->count(str) == 0) return false;
this->value = (*const_cast<t_config_enum_values*>(this->keys_map))[str];
return true;
};
2013-12-21 13:27:58 +00:00
};
enum ConfigOptionType {
coNone,
coFloat,
2013-12-21 13:27:58 +00:00
coFloats,
coInt,
2013-12-21 13:27:58 +00:00
coInts,
coString,
coStrings,
2014-03-22 15:23:33 +00:00
coPercent,
2013-12-20 15:37:28 +00:00
coFloatOrPercent,
2013-12-20 19:54:11 +00:00
coPoint,
2013-12-21 13:27:58 +00:00
coPoints,
2013-12-20 20:32:18 +00:00
coBool,
2013-12-21 13:27:58 +00:00
coBools,
coEnum,
};
class ConfigOptionDef
{
public:
ConfigOptionType type;
ConfigOption* default_value;
std::string gui_type;
std::string gui_flags;
std::string label;
2014-01-02 21:06:58 +00:00
std::string full_label;
std::string category;
std::string tooltip;
std::string sidetext;
std::string cli;
t_config_option_key ratio_over;
bool multiline;
bool full_width;
bool readonly;
int height;
int width;
int min;
int max;
std::vector<t_config_option_key> aliases;
std::vector<t_config_option_key> shortcut;
std::vector<std::string> enum_values;
std::vector<std::string> enum_labels;
t_config_enum_values enum_keys_map;
ConfigOptionDef() : type(coNone), default_value(NULL),
multiline(false), full_width(false), readonly(false),
height(-1), width(-1), min(INT_MIN), max(INT_MAX) {};
};
typedef std::map<t_config_option_key,ConfigOptionDef> t_optiondef_map;
class ConfigDef
{
public:
t_optiondef_map options;
~ConfigDef();
ConfigOptionDef* add(const t_config_option_key &opt_key, ConfigOptionType type);
const ConfigOptionDef* get(const t_config_option_key &opt_key) const;
};
class ConfigBase
{
public:
const ConfigDef* def;
ConfigBase() : def(NULL) {};
2015-12-16 11:33:19 +00:00
virtual ~ConfigBase() {};
2015-08-11 23:03:43 +00:00
bool has(const t_config_option_key &opt_key);
2015-12-16 11:33:19 +00:00
const ConfigOption* option(const t_config_option_key &opt_key) const;
ConfigOption* option(const t_config_option_key &opt_key, bool create = false);
virtual ConfigOption* optptr(const t_config_option_key &opt_key, bool create = false) = 0;
2015-07-01 16:18:25 +00:00
virtual t_config_option_keys keys() const = 0;
void apply(const ConfigBase &other, bool ignore_nonexistent = false);
bool equals(ConfigBase &other);
t_config_option_keys diff(ConfigBase &other);
2015-12-02 18:32:57 +00:00
std::string serialize(const t_config_option_key &opt_key) const;
2015-08-11 23:03:43 +00:00
bool set_deserialize(const t_config_option_key &opt_key, std::string str);
double get_abs_value(const t_config_option_key &opt_key);
double get_abs_value(const t_config_option_key &opt_key, double ratio_over);
2015-07-01 16:18:25 +00:00
void setenv_();
};
class DynamicConfig : public virtual ConfigBase
{
public:
DynamicConfig() {};
DynamicConfig(const DynamicConfig& other);
2014-05-08 11:33:43 +00:00
DynamicConfig& operator= (DynamicConfig other);
void swap(DynamicConfig &other);
2015-12-16 11:33:19 +00:00
virtual ~DynamicConfig();
2015-08-11 23:03:43 +00:00
template<class T> T* opt(const t_config_option_key &opt_key, bool create = false);
2015-12-16 11:33:19 +00:00
virtual ConfigOption* optptr(const t_config_option_key &opt_key, bool create = false);
2015-07-01 16:18:25 +00:00
t_config_option_keys keys() const;
2015-08-11 23:03:43 +00:00
void erase(const t_config_option_key &opt_key);
2013-12-20 15:37:28 +00:00
private:
typedef std::map<t_config_option_key,ConfigOption*> t_options_map;
t_options_map options;
};
class StaticConfig : public virtual ConfigBase
{
public:
StaticConfig() : ConfigBase() {};
2015-07-01 16:18:25 +00:00
t_config_option_keys keys() const;
2015-12-16 11:33:19 +00:00
//virtual ConfigOption* optptr(const t_config_option_key &opt_key, bool create = false) = 0;
void set_defaults();
};
}
#endif