2013-12-07 15:14:30 +00:00
|
|
|
#ifndef slic3r_Config_hpp_
|
|
|
|
#define slic3r_Config_hpp_
|
|
|
|
|
|
|
|
#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
|
|
|
|
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.
|
2013-12-07 15:14:30 +00:00
|
|
|
typedef std::string t_config_option_key;
|
|
|
|
typedef std::vector<std::string> t_config_option_keys;
|
|
|
|
|
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 {
|
|
|
|
public:
|
|
|
|
virtual ~ConfigOption() {};
|
2014-01-05 12:16:13 +00:00
|
|
|
virtual std::string serialize() const = 0;
|
2014-03-24 00:07:30 +00:00
|
|
|
virtual bool deserialize(std::string str) = 0;
|
2015-12-07 18:39:49 +00:00
|
|
|
virtual void set(const ConfigOption &option) = 0;
|
2014-03-25 23:08:15 +00:00
|
|
|
virtual int getInt() const { return 0; };
|
2015-12-07 18:39:49 +00:00
|
|
|
virtual double getFloat() const { return 0; };
|
|
|
|
virtual bool getBool() const { return false; };
|
2014-03-25 23:08:15 +00:00
|
|
|
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
|
|
|
};
|
|
|
|
|
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 {
|
|
|
|
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;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
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 {
|
|
|
|
public:
|
|
|
|
virtual ~ConfigOptionVectorBase() {};
|
|
|
|
virtual std::vector<std::string> vserialize() const = 0;
|
|
|
|
};
|
|
|
|
|
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
|
|
|
{
|
|
|
|
public:
|
|
|
|
virtual ~ConfigOptionVector() {};
|
|
|
|
std::vector<T> values;
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
void set(const ConfigOption &option) {
|
2015-12-08 09:53:57 +00:00
|
|
|
const ConfigOptionVector<T>* other = dynamic_cast< const ConfigOptionVector<T>* >(&option);
|
2015-12-07 18:39:49 +00:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionFloat : public ConfigOptionSingle<double>
|
2013-12-07 15:14:30 +00:00
|
|
|
{
|
|
|
|
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
|
|
|
|
2015-12-07 18:39:49 +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();
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
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();
|
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;
|
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();
|
|
|
|
};
|
|
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
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, ',')) {
|
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;
|
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
|
|
|
{
|
|
|
|
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
|
|
|
|
2014-03-25 23:08:15 +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();
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
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();
|
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;
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
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, ',')) {
|
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;
|
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
|
|
|
{
|
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
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;
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2013-12-20 19:54:11 +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
|
|
|
{
|
|
|
|
public:
|
|
|
|
|
2014-01-05 12:16:13 +00:00
|
|
|
std::string serialize() const {
|
2013-12-21 23:39:03 +00:00
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
2013-12-21 23:39:03 +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);
|
|
|
|
}
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
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
|
|
|
{
|
|
|
|
public:
|
2015-12-07 18:39:49 +00:00
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
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();
|
2014-03-22 15:23:33 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
class ConfigOptionFloatOrPercent : public ConfigOptionPercent
|
2013-12-20 15:37:28 +00:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
bool percent;
|
2015-12-07 18:39:49 +00:00
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
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();
|
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
|
|
|
{
|
|
|
|
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;
|
2015-12-07 18:39:49 +00:00
|
|
|
ss << this->value.x;
|
2013-12-20 19:54:11 +00:00
|
|
|
ss << ",";
|
2015-12-07 18:39:49 +00:00
|
|
|
ss << this->value.y;
|
2013-12-20 19:54:11 +00:00
|
|
|
return ss.str();
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
2014-08-04 09:34:53 +00:00
|
|
|
std::istringstream iss(str);
|
2015-12-07 18:39:49 +00:00
|
|
|
iss >> this->value.x;
|
2014-08-04 09:34:53 +00:00
|
|
|
iss.ignore(std::numeric_limits<std::streamsize>::max(), ',');
|
|
|
|
iss.ignore(std::numeric_limits<std::streamsize>::max(), 'x');
|
2015-12-07 18:39:49 +00:00
|
|
|
iss >> this->value.y;
|
2014-08-04 09:34:53 +00:00
|
|
|
return true;
|
2013-12-20 19:54:11 +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
|
|
|
{
|
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
2015-01-28 15:08:50 +00:00
|
|
|
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;
|
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')) {
|
|
|
|
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
|
|
|
}
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
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
|
|
|
{
|
|
|
|
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
|
|
|
|
2015-12-07 18:39:49 +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");
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
2013-12-20 20:32:18 +00:00
|
|
|
this->value = (str.compare("1") == 0);
|
2014-03-24 00:07:30 +00:00
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
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);
|
|
|
|
}
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2013-12-21 13:27:58 +00:00
|
|
|
};
|
2013-12-20 20:32:18 +00:00
|
|
|
};
|
|
|
|
|
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
|
|
|
{
|
|
|
|
public:
|
2015-12-07 18:39:49 +00:00
|
|
|
// 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 {
|
2013-12-21 15:15:41 +00:00
|
|
|
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;
|
2013-12-21 13:48:25 +00:00
|
|
|
}
|
|
|
|
return "";
|
|
|
|
};
|
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
2013-12-21 15:15:41 +00:00
|
|
|
t_config_enum_values enum_keys_map = ConfigOptionEnum<T>::get_enum_values();
|
2014-03-24 00:07:30 +00:00
|
|
|
if (enum_keys_map.count(str) == 0) return false;
|
2013-12-21 15:15:41 +00:00
|
|
|
this->value = static_cast<T>(enum_keys_map[str]);
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2013-12-21 13:48:25 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Map from an enum name to an enum integer value.
|
|
|
|
//FIXME The map is called often, it shall be initialized statically.
|
2013-12-21 15:15:41 +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
|
|
|
{
|
|
|
|
public:
|
2015-12-07 18:39:49 +00:00
|
|
|
const t_config_enum_values* keys_map;
|
2013-12-21 15:15:41 +00:00
|
|
|
|
2014-01-05 12:16:13 +00:00
|
|
|
std::string serialize() const {
|
2015-12-07 18:39:49 +00:00
|
|
|
for (t_config_enum_values::const_iterator it = this->keys_map->begin(); it != this->keys_map->end(); ++it) {
|
2013-12-21 15:15:41 +00:00
|
|
|
if (it->second == this->value) return it->first;
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
};
|
2013-12-20 20:32:18 +00:00
|
|
|
|
2014-03-24 00:07:30 +00:00
|
|
|
bool deserialize(std::string str) {
|
|
|
|
if (this->keys_map->count(str) == 0) return false;
|
2015-12-07 18:39:49 +00:00
|
|
|
this->value = (*const_cast<t_config_enum_values*>(this->keys_map))[str];
|
2014-03-24 00:07:30 +00:00
|
|
|
return true;
|
2013-12-21 15:15:41 +00:00
|
|
|
};
|
2013-12-21 13:27:58 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// Type of a configuration value.
|
2013-12-07 15:14:30 +00:00
|
|
|
enum ConfigOptionType {
|
2015-11-04 19:48:44 +00:00
|
|
|
coNone,
|
2016-09-13 11:30:00 +00:00
|
|
|
// single float
|
2013-12-07 15:14:30 +00:00
|
|
|
coFloat,
|
2016-09-13 11:30:00 +00:00
|
|
|
// vector of floats
|
2013-12-21 13:27:58 +00:00
|
|
|
coFloats,
|
2016-09-13 11:30:00 +00:00
|
|
|
// single int
|
2013-12-07 15:14:30 +00:00
|
|
|
coInt,
|
2016-09-13 11:30:00 +00:00
|
|
|
// vector of ints
|
2013-12-21 13:27:58 +00:00
|
|
|
coInts,
|
2016-09-13 11:30:00 +00:00
|
|
|
// single string
|
2013-12-07 15:14:30 +00:00
|
|
|
coString,
|
2016-09-13 11:30:00 +00:00
|
|
|
// vector of strings
|
2013-12-21 23:39:03 +00:00
|
|
|
coStrings,
|
2016-09-13 11:30:00 +00:00
|
|
|
// percent value. Currently only used for infill.
|
2014-03-22 15:23:33 +00:00
|
|
|
coPercent,
|
2016-09-13 11:30:00 +00:00
|
|
|
// a fraction or an absolute value
|
2013-12-20 15:37:28 +00:00
|
|
|
coFloatOrPercent,
|
2016-09-13 11:30:00 +00:00
|
|
|
// single 2d point. Currently not used.
|
2013-12-20 19:54:11 +00:00
|
|
|
coPoint,
|
2016-09-13 11:30:00 +00:00
|
|
|
// vector of 2d points. Currently used for the definition of the print bed and for the extruder offsets.
|
2013-12-21 13:27:58 +00:00
|
|
|
coPoints,
|
2016-09-13 11:30:00 +00:00
|
|
|
// single boolean value
|
2013-12-20 20:32:18 +00:00
|
|
|
coBool,
|
2016-09-13 11:30:00 +00:00
|
|
|
// vector of boolean values
|
2013-12-21 13:27:58 +00:00
|
|
|
coBools,
|
2016-09-13 11:30:00 +00:00
|
|
|
// a generic enum
|
2013-12-21 15:15:41 +00:00
|
|
|
coEnum,
|
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
|
|
|
|
{
|
|
|
|
public:
|
2016-09-13 11:30:00 +00:00
|
|
|
// What type? bool, int, string etc.
|
2013-12-07 15:14:30 +00:00
|
|
|
ConfigOptionType type;
|
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.
|
2015-12-07 18:39:49 +00:00
|
|
|
ConfigOption* default_value;
|
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).
|
2014-06-19 18:07:16 +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.
|
2014-06-19 18:07:16 +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.
|
2013-12-07 15:14:30 +00:00
|
|
|
std::string label;
|
2014-01-02 21:06:58 +00:00
|
|
|
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"
|
2013-12-21 20:06:45 +00:00
|
|
|
std::string category;
|
2016-09-13 11:30:00 +00:00
|
|
|
// A tooltip text shown in the GUI.
|
2013-12-07 15:14:30 +00:00
|
|
|
std::string tooltip;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Text right from the input field, usually a unit of measurement.
|
2013-12-21 20:06:45 +00:00
|
|
|
std::string sidetext;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Format of this parameter on a command line.
|
2013-12-21 20:06:45 +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.
|
2013-12-21 20:06:45 +00:00
|
|
|
t_config_option_key ratio_over;
|
2016-09-13 11:30:00 +00:00
|
|
|
// True for multiline strings.
|
2013-12-21 20:06:45 +00:00
|
|
|
bool multiline;
|
2016-09-13 11:30:00 +00:00
|
|
|
// For text input: If true, the GUI text box spans the complete page width.
|
2013-12-21 20:06:45 +00:00
|
|
|
bool full_width;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Not editable. Currently only used for the display of the number of threads.
|
2013-12-21 20:06:45 +00:00
|
|
|
bool readonly;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Height of a multiline GUI text box.
|
2013-12-21 20:06:45 +00:00
|
|
|
int height;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Optional width of an input field.
|
2013-12-21 20:06:45 +00:00
|
|
|
int width;
|
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.
|
2013-12-21 20:06:45 +00:00
|
|
|
int min;
|
|
|
|
int max;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Legacy names for this configuration option.
|
|
|
|
// Used when parsing legacy configuration file.
|
2013-12-21 20:06:45 +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".
|
2013-12-21 20:06:45 +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".
|
2013-12-21 20:06:45 +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()
|
2013-12-21 15:15:41 +00:00
|
|
|
t_config_enum_values enum_keys_map;
|
2016-09-13 11:30:00 +00:00
|
|
|
|
2015-12-07 18:39:49 +00:00
|
|
|
ConfigOptionDef() : type(coNone), default_value(NULL),
|
2015-11-04 19:48:44 +00:00
|
|
|
multiline(false), full_width(false), readonly(false),
|
2013-12-21 23:39:03 +00:00
|
|
|
height(-1), width(-1), min(INT_MIN), max(INT_MAX) {};
|
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
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
t_optiondef_map options;
|
|
|
|
~ConfigDef();
|
2015-12-16 12:09:25 +00:00
|
|
|
ConfigOptionDef* add(const t_config_option_key &opt_key, ConfigOptionType type);
|
2015-12-07 18:39:49 +00:00
|
|
|
const ConfigOptionDef* get(const t_config_option_key &opt_key) const;
|
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +00:00
|
|
|
// An abstract configuration store.
|
2013-12-07 15:14:30 +00:00
|
|
|
class ConfigBase
|
|
|
|
{
|
|
|
|
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.
|
|
|
|
// ConfigBase does not own ConfigDef, it only references it.
|
2015-12-07 18:39:49 +00:00
|
|
|
const ConfigDef* def;
|
2013-12-21 15:15:41 +00:00
|
|
|
|
|
|
|
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;
|
2014-03-22 19:12:54 +00:00
|
|
|
void apply(const ConfigBase &other, bool ignore_nonexistent = false);
|
2014-11-09 11:25:59 +00:00
|
|
|
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);
|
2016-09-13 11:30:00 +00:00
|
|
|
|
2015-08-11 23:03:43 +00:00
|
|
|
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_();
|
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
|
|
|
{
|
|
|
|
public:
|
2013-12-21 20:06:45 +00:00
|
|
|
DynamicConfig() {};
|
2014-03-22 19:12:54 +00:00
|
|
|
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-07 15:14:30 +00:00
|
|
|
|
2013-12-20 15:37:28 +00:00
|
|
|
private:
|
|
|
|
typedef std::map<t_config_option_key,ConfigOption*> t_options_map;
|
|
|
|
t_options_map options;
|
2013-12-07 15:14:30 +00:00
|
|
|
};
|
|
|
|
|
2016-09-13 11:30:00 +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
|
|
|
{
|
|
|
|
public:
|
2015-12-07 18:39:49 +00:00
|
|
|
StaticConfig() : ConfigBase() {};
|
2016-09-13 11:30:00 +00:00
|
|
|
// 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;
|
2016-09-13 11:30:00 +00:00
|
|
|
// Set all statically defined config options to their defaults defined by this->def.
|
2015-12-07 18:39:49 +00:00
|
|
|
void set_defaults();
|
2016-09-13 11:30:00 +00:00
|
|
|
// The derived class has to implement optptr to resolve a static configuration value.
|
|
|
|
// virtual ConfigOption* optptr(const t_config_option_key &opt_key, bool create = false) = 0;
|
2013-12-07 15:14:30 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|