61 std::unique_ptr< DiscreteVariable >
fastVariable(std::string var_description,
62 const std::string& default_domain) {
66 std::vector< std::string > labels;
67 std::vector< GUM_SCALAR > ticks;
70 std::vector< int > values;
71 std::vector< double > numerical_values;
73 trim(var_description);
77 auto t = *default_domain.begin();
78 if (t !=
'[' && t !=
'{')
80 "default_domain (" << default_domain <<
") must start with '[' or '{'")
82 if (var_description.find(
'[') == std::string::npos
83 && var_description.find(
'{') == std::string::npos) {
84 var_description += default_domain;
88 if (*(var_description.rbegin()) ==
']') {
89 auto posBrack = var_description.find(
'[');
90 if (posBrack != std::string::npos) {
91 name = var_description.substr(0, posBrack);
93 = var_description.substr(posBrack + 1, var_description.size() - posBrack - 2);
94 const auto& args =
split(s_args,
",");
98 }
else if (args.size() == 1) {
101 const auto& labels =
split(args[0],
":");
103 if (labels.size() == 3) {
105 const double fmin = std::stod(labels[0]);
106 const double fmax = std::stod(labels[1]);
107 const int nbr = std::stoi(labels[2]);
111 const double step =
double((fmax - fmin) / nbr);
112 double current = fmin;
113 for (
auto i = 0; i <= nbr; i += 1) {
114 ticks.push_back(current);
120 int n = std::stoi(args[0]);
122 if (default_domain !=
"[1]")
124 ds =
static_cast< Size >(n);
126 range_max = long(ds) - 1;
128 }
else if (args.size() == 2) {
130 range_min = std::stol(args[0]);
131 range_max = std::stol(args[1]);
133 if (range_max < range_min)
135 "Invalid range for variable " << var_description <<
": max<min")
136 if (range_max == range_min && default_domain !=
"[1]")
138 "Invalid range for variable "
139 << var_description <<
": max==min not allowed if default_domain_size>1")
141 ds =
static_cast< Size >(1 + range_max - range_min);
144 for (
const auto& tick: args) {
145 ticks.push_back(
static_cast< GUM_SCALAR
>(std::strtod(tick.c_str(),
nullptr)));
147 ds = args.size() - 1;
151 }
else if (*(var_description.rbegin()) ==
'}') {
152 auto posBrack = var_description.find(
'{');
153 if (posBrack != std::string::npos) {
154 name = var_description.substr(0, posBrack);
155 labels =
split(var_description.substr(posBrack + 1, var_description.size() - posBrack - 2),
157 if (labels.size() == 3) {
159 const auto fmin = std::stod(labels[0]);
160 const auto fmax = std::stod(labels[1]);
161 const int nbr = std::stoi(labels[2]);
165 const double step =
double((fmax - fmin) / (nbr - 1));
167 if ((trunc(step) == step) && (trunc(fmin) == fmin) && (trunc(fmax) == fmax)) {
170 numerical_values.clear();
172 for (
int i = 1; i <= nbr; i++) {
173 labels.push_back(std::to_string(v));
180 numerical_values.clear();
183 numerical_values.push_back(v);
184 for (
auto i = 1; i < nbr - 1; i++) {
186 numerical_values.push_back(v);
188 numerical_values.push_back(fmax);
192 =
split(var_description.substr(posBrack + 1, var_description.size() - posBrack - 2),
194 if (labels.size() < 2) {
195 if (labels.size() == 1
196 && default_domain !=
"[1]")
199 if (!hasUniqueElts(labels)) {
206 name = var_description;
211 }
else if (ds == 1) {
212 if (default_domain !=
"[1]")
214 "Only one value for variable " << var_description <<
" (2 at least are needed).")
217 if (!labels.empty()) {
218 if (std::all_of(labels.cbegin(), labels.cend(),
isInteger)) {
219 for (
const auto& label: labels)
220 values.push_back(std::stoi(label));
221 if (values.size() >= 2) {
223 std::sort(values.begin(), values.end());
224 range_min = values[0];
226 bool is_range =
true;
227 for (
const auto item: values) {
228 if (item != v) { is_range =
false; }
237 }
else if (std::all_of(labels.cbegin(), labels.cend(),
isNumerical))
238 for (
const auto& label: labels)
239 numerical_values.push_back(std::stod(label));
244 if (!values.empty()) {
245 return std::make_unique< IntegerVariable >(name, name, values);
246 }
else if (!numerical_values.empty()) {
247 if (!std::all_of(numerical_values.cbegin(),
248 numerical_values.cend(),
252 return std::make_unique< NumericalDiscreteVariable >(name, name, numerical_values);
253 }
else if (!labels.empty()) {
254 return std::make_unique< LabelizedVariable >(name, name, labels);
255 }
else if (!ticks.empty()) {
261 if (name.back() ==
'+') {
264 return std::make_unique< DiscretizedVariable< GUM_SCALAR > >(name, name, ticks,
true);
266 return std::make_unique< DiscretizedVariable< GUM_SCALAR > >(name, name, ticks);
269 return std::make_unique< RangeVariable >(name, name, range_min, range_max);