2017-12-19 10:59:42 +00:00
|
|
|
|
#include "OptionsGroup.hpp"
|
2017-12-13 13:45:10 +00:00
|
|
|
|
#include "ConfigExceptions.hpp"
|
2017-12-05 14:54:01 +00:00
|
|
|
|
|
2017-12-13 13:45:10 +00:00
|
|
|
|
#include <utility>
|
|
|
|
|
#include <wx/tooltip.h>
|
2017-12-05 14:54:01 +00:00
|
|
|
|
|
|
|
|
|
namespace Slic3r { namespace GUI {
|
|
|
|
|
|
2017-12-13 13:45:10 +00:00
|
|
|
|
const t_field& OptionsGroup::build_field(const Option& opt) {
|
|
|
|
|
return build_field(opt.opt_id, opt.opt);
|
|
|
|
|
}
|
|
|
|
|
const t_field& OptionsGroup::build_field(const t_config_option_key& id) {
|
2018-01-05 14:11:33 +00:00
|
|
|
|
const ConfigOptionDef& opt = m_options_map.at(id);
|
2017-12-13 13:45:10 +00:00
|
|
|
|
return build_field(id, opt);
|
|
|
|
|
}
|
2017-12-05 14:54:01 +00:00
|
|
|
|
|
2017-12-13 13:45:10 +00:00
|
|
|
|
const t_field& OptionsGroup::build_field(const t_config_option_key& id, const ConfigOptionDef& opt) {
|
|
|
|
|
// Check the gui_type field first, fall through
|
|
|
|
|
// is the normal type.
|
|
|
|
|
if (opt.gui_type.compare("select") == 0) {
|
|
|
|
|
} else if (opt.gui_type.compare("select_open") == 0) {
|
2018-01-05 14:11:33 +00:00
|
|
|
|
m_fields.emplace(id, STDMOVE(Choice::Create<Choice>(m_parent, opt, id)));
|
2017-12-13 13:45:10 +00:00
|
|
|
|
} else if (opt.gui_type.compare("color") == 0) {
|
2018-01-05 14:11:33 +00:00
|
|
|
|
m_fields.emplace(id, STDMOVE(ColourPicker::Create<ColourPicker>(m_parent, opt, id)));
|
2017-12-13 13:45:10 +00:00
|
|
|
|
} else if (opt.gui_type.compare("f_enum_open") == 0 ||
|
|
|
|
|
opt.gui_type.compare("i_enum_open") == 0 ||
|
|
|
|
|
opt.gui_type.compare("i_enum_closed") == 0) {
|
2018-01-05 14:11:33 +00:00
|
|
|
|
m_fields.emplace(id, STDMOVE(Choice::Create<Choice>(m_parent, opt, id)));
|
2017-12-13 13:45:10 +00:00
|
|
|
|
} else if (opt.gui_type.compare("slider") == 0) {
|
|
|
|
|
} else if (opt.gui_type.compare("i_spin") == 0) { // Spinctrl
|
|
|
|
|
} else {
|
|
|
|
|
switch (opt.type) {
|
|
|
|
|
case coFloatOrPercent:
|
|
|
|
|
case coFloat:
|
2017-12-22 10:50:28 +00:00
|
|
|
|
case coFloats:
|
2017-12-18 12:58:51 +00:00
|
|
|
|
case coPercent:
|
2017-12-22 10:50:28 +00:00
|
|
|
|
case coPercents:
|
2017-12-18 12:58:51 +00:00
|
|
|
|
case coString:
|
|
|
|
|
case coStrings:
|
2018-01-05 14:11:33 +00:00
|
|
|
|
m_fields.emplace(id, STDMOVE(TextCtrl::Create<TextCtrl>(m_parent, opt, id)));
|
2017-12-13 13:45:10 +00:00
|
|
|
|
break;
|
2017-12-18 12:58:51 +00:00
|
|
|
|
case coBool:
|
|
|
|
|
case coBools:
|
2018-01-05 14:11:33 +00:00
|
|
|
|
m_fields.emplace(id, STDMOVE(CheckBox::Create<CheckBox>(m_parent, opt, id)));
|
2017-12-18 12:58:51 +00:00
|
|
|
|
break;
|
|
|
|
|
case coInt:
|
|
|
|
|
case coInts:
|
2018-01-05 14:11:33 +00:00
|
|
|
|
m_fields.emplace(id, STDMOVE(SpinCtrl::Create<SpinCtrl>(m_parent, opt, id)));
|
2017-12-18 12:58:51 +00:00
|
|
|
|
break;
|
|
|
|
|
case coEnum:
|
2018-01-05 14:11:33 +00:00
|
|
|
|
m_fields.emplace(id, STDMOVE(Choice::Create<Choice>(m_parent, opt, id)));
|
2017-12-18 12:58:51 +00:00
|
|
|
|
break;
|
2017-12-22 10:50:28 +00:00
|
|
|
|
case coPoints:
|
2018-01-05 14:11:33 +00:00
|
|
|
|
m_fields.emplace(id, STDMOVE(Point::Create<Point>(m_parent, opt, id)));
|
2017-12-22 10:50:28 +00:00
|
|
|
|
break;
|
2017-12-13 13:45:10 +00:00
|
|
|
|
case coNone: break;
|
|
|
|
|
default:
|
2017-12-18 12:58:51 +00:00
|
|
|
|
throw /*//!ConfigGUITypeError("")*/std::logic_error("This control doesn't exist till now"); break;
|
2017-12-13 13:45:10 +00:00
|
|
|
|
}
|
2017-12-05 14:54:01 +00:00
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
// Grab a reference to fields for convenience
|
2018-01-05 14:11:33 +00:00
|
|
|
|
const t_field& field = m_fields[id];
|
|
|
|
|
field->m_on_change = [this](std::string opt_id, boost::any value){
|
|
|
|
|
//! This function will be called from Field.
|
|
|
|
|
//! Call OptionGroup._on_change(...)
|
|
|
|
|
if (!this->m_disabled)
|
|
|
|
|
this->on_change_OG(opt_id, value);
|
|
|
|
|
};
|
|
|
|
|
field->m_parent = parent();
|
2017-12-13 13:45:10 +00:00
|
|
|
|
// assign function objects for callbacks, etc.
|
|
|
|
|
return field;
|
2017-12-05 14:54:01 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void OptionsGroup::append_line(const Line& line) {
|
2017-12-22 10:50:28 +00:00
|
|
|
|
//! if (line.sizer != nullptr || (line.widget != nullptr && line.full_width > 0)){
|
|
|
|
|
if ( (line.sizer != nullptr || line.widget != nullptr) && line.full_width){
|
|
|
|
|
if (line.sizer != nullptr) {
|
2017-12-13 13:45:10 +00:00
|
|
|
|
sizer->Add(line.sizer, 0, wxEXPAND | wxALL, wxOSX ? 0 : 15);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (line.widget != nullptr) {
|
2018-01-05 14:11:33 +00:00
|
|
|
|
sizer->Add(line.widget(m_parent), 0, wxEXPAND | wxALL, wxOSX ? 0 : 15);
|
2017-12-13 13:45:10 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
2017-12-05 14:54:01 +00:00
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
|
2017-12-22 10:50:28 +00:00
|
|
|
|
auto option_set = line.get_options();
|
|
|
|
|
|
|
|
|
|
// if we have a single option with no label, no sidetext just add it directly to sizer
|
|
|
|
|
if (option_set.size() == 1 && label_width == 0 && option_set.front().opt.full_width &&
|
|
|
|
|
option_set.front().opt.sidetext.size() == 0 && option_set.front().side_widget == nullptr &&
|
|
|
|
|
line.get_extra_widgets().size() == 0) {
|
|
|
|
|
const auto& option = option_set.front();
|
|
|
|
|
const auto& field = build_field(option);
|
|
|
|
|
|
|
|
|
|
if (is_window_field(field))
|
|
|
|
|
sizer->Add(field->getWindow(), 0, wxEXPAND | wxALL, wxOSX ? 0 : 5);
|
|
|
|
|
if (is_sizer_field(field))
|
|
|
|
|
sizer->Add(field->getSizer(), 0, wxEXPAND | wxALL, wxOSX ? 0 : 5);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
|
auto grid_sizer = m_grid_sizer;
|
2017-12-13 13:45:10 +00:00
|
|
|
|
|
|
|
|
|
// Build a label if we have it
|
2017-12-05 14:54:01 +00:00
|
|
|
|
if (label_width != 0) {
|
2017-12-14 13:42:47 +00:00
|
|
|
|
auto label = new wxStaticText(parent(), wxID_ANY, line.label + (line.label.IsEmpty() ? "" : ":" ), wxDefaultPosition, wxSize(label_width, -1));
|
2017-12-13 13:45:10 +00:00
|
|
|
|
label->SetFont(label_font);
|
|
|
|
|
label->Wrap(label_width); // avoid a Linux/GTK bug
|
|
|
|
|
grid_sizer->Add(label, 0, wxALIGN_CENTER_VERTICAL,0);
|
|
|
|
|
if (line.label_tooltip.compare("") != 0)
|
|
|
|
|
label->SetToolTip(line.label_tooltip);
|
2017-12-05 14:54:01 +00:00
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
|
|
|
|
|
// If there's a widget, build it and add the result to the sizer.
|
|
|
|
|
if (line.widget != nullptr) {
|
|
|
|
|
auto wgt = line.widget(parent());
|
|
|
|
|
grid_sizer->Add(wgt, 0, wxEXPAND | wxALL, wxOSX ? 0 : 15);
|
2017-12-05 14:54:01 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2017-12-13 13:45:10 +00:00
|
|
|
|
|
|
|
|
|
// if we have a single option with no sidetext just add it directly to the grid sizer
|
2017-12-22 10:50:28 +00:00
|
|
|
|
//! auto option_set = line.get_options();
|
2017-12-13 13:45:10 +00:00
|
|
|
|
if (option_set.size() == 1 && option_set.front().opt.sidetext.size() == 0 &&
|
|
|
|
|
option_set.front().side_widget == nullptr && line.get_extra_widgets().size() == 0) {
|
|
|
|
|
const auto& option = option_set.front();
|
2017-12-18 12:58:51 +00:00
|
|
|
|
const auto& field = build_field(option);
|
|
|
|
|
//! std::cerr << "single option, no sidetext.\n";
|
|
|
|
|
//! std::cerr << "field parent is not null?: " << (field->parent != nullptr) << "\n";
|
|
|
|
|
|
|
|
|
|
if (is_window_field(field))
|
|
|
|
|
grid_sizer->Add(field->getWindow(), 0, (option.opt.full_width ? wxEXPAND : 0) | wxALIGN_CENTER_VERTICAL, 0);
|
|
|
|
|
if (is_sizer_field(field))
|
|
|
|
|
grid_sizer->Add(field->getSizer(), 0, (option.opt.full_width ? wxEXPAND : 0) | wxALIGN_CENTER_VERTICAL, 0);
|
2017-12-13 13:45:10 +00:00
|
|
|
|
return;
|
2017-12-05 14:54:01 +00:00
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
|
|
|
|
|
// if we're here, we have more than one option or a single option with sidetext
|
|
|
|
|
// so we need a horizontal sizer to arrange these things
|
|
|
|
|
auto sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
|
grid_sizer->Add(sizer, 0, 0, 0);
|
2017-12-14 13:42:47 +00:00
|
|
|
|
for (auto opt : option_set) {
|
|
|
|
|
ConfigOptionDef option = opt.opt;
|
|
|
|
|
// add label if any
|
|
|
|
|
if (option.label != "") {
|
|
|
|
|
auto field_label = new wxStaticText(parent(), wxID_ANY, wxString(option.label) + ":", wxDefaultPosition, wxDefaultSize);
|
|
|
|
|
field_label->SetFont(sidetext_font);
|
|
|
|
|
sizer->Add(field_label, 0, wxALIGN_CENTER_VERTICAL, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// add field
|
2017-12-18 12:58:51 +00:00
|
|
|
|
const Option& opt_ref = opt;
|
2017-12-22 10:50:28 +00:00
|
|
|
|
auto& field = build_field(opt_ref);
|
|
|
|
|
is_sizer_field(field) ?
|
|
|
|
|
sizer->Add(field->getSizer(), 0, wxALIGN_CENTER_VERTICAL, 0) :
|
|
|
|
|
sizer->Add(field->getWindow(), 0, wxALIGN_CENTER_VERTICAL, 0);
|
2017-12-14 13:42:47 +00:00
|
|
|
|
|
|
|
|
|
// add sidetext if any
|
|
|
|
|
if (option.sidetext != "") {
|
2017-12-19 10:59:42 +00:00
|
|
|
|
auto sidetext = new wxStaticText(parent(), wxID_ANY, wxString::FromUTF8(option.sidetext.c_str()), wxDefaultPosition, wxDefaultSize);
|
2017-12-14 13:42:47 +00:00
|
|
|
|
sidetext->SetFont(sidetext_font);
|
|
|
|
|
sizer->Add(sidetext, 0, wxLEFT | wxALIGN_CENTER_VERTICAL, 4);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// add side widget if any
|
|
|
|
|
if (opt.side_widget != nullptr) {
|
2017-12-22 10:50:28 +00:00
|
|
|
|
sizer->Add(opt.side_widget(parent())/*!.target<wxWindow>()*/, 0, wxLEFT | wxALIGN_CENTER_VERTICAL, 1); //! requires verification
|
2017-12-14 13:42:47 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (opt.opt_id != option_set.back().opt_id) //! istead of (opt != option_set.back())
|
|
|
|
|
{
|
|
|
|
|
sizer->AddSpacer(4);
|
|
|
|
|
}
|
2017-12-22 10:50:28 +00:00
|
|
|
|
}
|
|
|
|
|
// add extra sizers if any
|
|
|
|
|
for (auto extra_widget : line.get_extra_widgets()) {
|
|
|
|
|
sizer->Add(extra_widget(parent())/*!.target<wxWindow>()*/, 0, wxLEFT | wxALIGN_CENTER_VERTICAL, 4); //! requires verification
|
2017-12-14 13:42:47 +00:00
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
}
|
2017-12-14 13:42:47 +00:00
|
|
|
|
|
2017-12-13 13:45:10 +00:00
|
|
|
|
Line OptionsGroup::create_single_option_line(const Option& option) const {
|
2017-12-22 10:50:28 +00:00
|
|
|
|
Line retval {option.opt.label, option.opt.tooltip};
|
2017-12-13 13:45:10 +00:00
|
|
|
|
Option tmp(option);
|
|
|
|
|
tmp.opt.label = std::string("");
|
|
|
|
|
retval.append_option(tmp);
|
|
|
|
|
return retval;
|
2017-12-05 14:54:01 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
|
void OptionsGroup::on_change_OG(t_config_option_key id, /*config_value*/boost::any value) {
|
|
|
|
|
if (m_on_change != nullptr)
|
|
|
|
|
m_on_change(id, value);
|
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
|
|
|
|
|
void OptionsGroup::_on_kill_focus (t_config_option_key id) {
|
|
|
|
|
// do nothing.
|
2017-12-05 14:54:01 +00:00
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
|
void ConfigOptionsGroup::on_change_OG(t_config_option_key opt_id, boost::any value)
|
|
|
|
|
{
|
|
|
|
|
// if (m_options_map.at(opt_id)/*exists $self->_opt_map->{$opt_id}*/) {
|
|
|
|
|
// my($opt_key, $opt_index) = @{ $self->_opt_map->{$opt_id} };
|
|
|
|
|
// auto option = m_options->{$opt_id};
|
|
|
|
|
|
|
|
|
|
// get value
|
|
|
|
|
auto field_value = get_value(opt_id);
|
|
|
|
|
// if ($option->gui_flags = ~/ \bserialized\b / ) {
|
|
|
|
|
// die "Can't set serialized option indexed value" if $opt_index != -1;
|
|
|
|
|
// # Split a string to multiple strings by a semi - colon.This is the old way of storing multi - string values.
|
|
|
|
|
// # Currently used for the post_process config value only.
|
|
|
|
|
// my @values = split / ; / , $field_value;
|
|
|
|
|
// $self->config->set($opt_key, \@values);
|
|
|
|
|
// }
|
|
|
|
|
// else {
|
|
|
|
|
// if ($opt_index == -1) {
|
|
|
|
|
// $self->config->set($opt_key, $field_value);
|
|
|
|
|
//! m_config->set_key_value(opt_id, new ConfigOption(value));
|
|
|
|
|
// }
|
|
|
|
|
// else {
|
|
|
|
|
// my $value = $self->config->get($opt_key);
|
|
|
|
|
// $value->[$opt_index] = $field_value;
|
|
|
|
|
// $self->config->set($opt_key, $value);
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
OptionsGroup::on_change_OG(opt_id, value);
|
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
|
} // GUI
|
|
|
|
|
} // Slic3r
|