#include "GUI.hpp"//"slic3r_gui.hpp" #include "Field.hpp" //#include #include #include #include "PrintConfig.hpp" namespace Slic3r { namespace GUI { void Field::_on_kill_focus(wxFocusEvent& event) { // Without this, there will be nasty focus bugs on Windows. // Also, docs for wxEvent::Skip() say "In general, it is recommended to skip all // non-command events to allow the default handling to take place." event.Skip(1); // call the registered function if it is available //! if (on_kill_focus) //! on_kill_focus(opt_id); } void Field::_on_change(wxCommandEvent& event) { std::cerr << "calling Field::_on_change \n"; //! if (on_change != nullptr && !disable_change_event) //! on_change(opt_id, "A"); } bool Field::is_matched(std::string string, std::string pattern) { std::regex regex_pattern(pattern, std::regex_constants::icase); // use ::icase to make the matching case insensitive like /i in perl return std::regex_match(string, regex_pattern); } void TextCtrl::BUILD() { auto size = wxSize(wxDefaultSize); if (opt.height >= 0) size.SetHeight(opt.height); if (opt.width >= 0) size.SetWidth(opt.width); wxString text_value = wxString(""); switch (opt.type) { case coFloatOrPercent: { if (static_cast(opt.default_value)->percent) { text_value = wxString::Format(_T("%i"), int(opt.default_value->getFloat())); text_value += "%"; } else wxNumberFormatter::ToString(opt.default_value->getFloat(), 2); break; } case coPercent: { text_value = wxString::Format(_T("%i"), int(opt.default_value->getFloat())); text_value += "%"; break; } case coFloat: { double val = opt.default_value->getFloat(); text_value = (val - int(val)) == 0 ? wxString::Format(_T("%i"), int(val)) : wxNumberFormatter::ToString(val, 2); break; } case coString: text_value += static_cast(opt.default_value)->value; break; case coStrings: { const ConfigOptionVectorBase *vec = static_cast(opt.default_value); if (vec == nullptr) break; if (vec->empty()) break; for (size_t id = 0; id < vec->size(); ++id) text_value += static_cast(opt.default_value)->value[id]; break; } default: break; } auto temp = new wxTextCtrl(parent, wxID_ANY, text_value, wxDefaultPosition, size, (opt.multiline ? wxTE_MULTILINE : 0)); if (opt.tooltip.length() > 0) { temp->SetToolTip(opt.tooltip); } temp->Bind(wxEVT_TEXT, ([=](wxCommandEvent e) { _on_change(e); }), temp->GetId()); temp->Bind(wxEVT_KILL_FOCUS, ([this](wxFocusEvent e) { _on_kill_focus(e); }), temp->GetId()); // recast as a wxWindow to fit the calling convention window = dynamic_cast(temp); } void TextCtrl::enable() { dynamic_cast(window)->Enable(); dynamic_cast(window)->SetEditable(true); } void TextCtrl::disable() { dynamic_cast(window)->Disable(); dynamic_cast(window)->SetEditable(false); } void TextCtrl::set_tooltip(const wxString& tip) { } void CheckBox::BUILD() { auto size = wxSize(wxDefaultSize); if (opt.height >= 0) size.SetHeight(opt.height); if (opt.width >= 0) size.SetWidth(opt.width); auto temp = new wxCheckBox(parent, wxID_ANY, wxString(""), wxDefaultPosition, size); temp->SetValue(opt.default_value->getBool()); if (opt.readonly) temp->Disable(); temp->Bind(wxEVT_CHECKBOX, ([this](wxCommandEvent e) { _on_change(e); }), temp->GetId()); if (opt.tooltip.length() > 0) { temp->SetToolTip(opt.tooltip); } // recast as a wxWindow to fit the calling convention window = dynamic_cast(temp); } int undef_spin_val = -9999; //! Probably, It's not necessary void SpinCtrl::BUILD() { auto size = wxSize(wxDefaultSize); if (opt.height >= 0) size.SetHeight(opt.height); if (opt.width >= 0) size.SetWidth(opt.width); auto temp = new wxSpinCtrl(parent, wxID_ANY, wxString::Format(_T("%i"), opt.default_value->getInt()), wxDefaultPosition, size, 0, opt.min >0 ? opt.min : 0, opt.max < 2147483647 ? opt.max : 2147483647, opt.default_value->getInt()); temp->Bind(wxEVT_SPINCTRL, ([=](wxCommandEvent e) { tmp_value = undef_spin_val; _on_change(e); }), temp->GetId()); temp->Bind(wxEVT_KILL_FOCUS, ([this](wxFocusEvent e) { tmp_value = undef_spin_val; _on_kill_focus(e); }), temp->GetId()); temp->Bind(wxEVT_TEXT, ([=](wxCommandEvent e) { // # On OSX / Cocoa, wxSpinCtrl::GetValue() doesn't return the new value // # when it was changed from the text control, so the on_change callback // # gets the old one, and on_kill_focus resets the control to the old value. // # As a workaround, we get the new value from $event->GetString and store // # here temporarily so that we can return it from $self->get_value std::string value = e.GetString(); if (is_matched(value, "^\d+$")) tmp_value = std::stoi(value); _on_change(e); // # We don't reset tmp_value here because _on_change might put callbacks // # in the CallAfter queue, and we want the tmp value to be available from // # them as well. }), temp->GetId()); if (opt.tooltip.length() > 0) { temp->SetToolTip(opt.tooltip); } // recast as a wxWindow to fit the calling convention window = dynamic_cast(temp); } void Choice::BUILD() { auto size = wxSize(wxDefaultSize); if (opt.height >= 0) size.SetHeight(opt.height); if (opt.width >= 0) size.SetWidth(opt.width); auto temp = new wxComboBox(parent, wxID_ANY, wxString(""), wxDefaultPosition, size); if (opt.gui_type.compare("") != 0 && opt.gui_type.compare("select_open") == 0) temp->SetWindowStyle(wxCB_READONLY); // // recast as a wxWindow to fit the calling convention window = dynamic_cast(temp); if (!opt.enum_labels.empty() && !opt.enum_values.empty()) { for (auto el : opt.enum_labels) temp->Append(wxString(el)); // temp->SetSelection(static_cast*>(opt.default_value)->value); //set_value(opt.default_value); set_selection(); } temp->Bind(wxEVT_TEXT, ([=](wxCommandEvent e) { _on_change(e); }), temp->GetId()); temp->Bind(wxEVT_COMBOBOX, ([this](wxCommandEvent e) { _on_change(e); }), temp->GetId()); if (opt.tooltip.length() > 0) temp->SetToolTip(opt.tooltip); } void Choice::set_selection() { wxString text_value = wxString(""); switch (opt.type){ case coFloat: case coPercent: { double val = opt.default_value->getFloat(); text_value = val - int(val) == 0 ? wxString::Format(_T("%i"), int(val)) : wxNumberFormatter::ToString(val, 1); auto idx = 0; for (auto el : opt.enum_values) { if (el.compare(text_value) == 0) break; ++idx; } if (opt.type == coPercent) text_value += "%"; idx == opt.enum_values.size() ? dynamic_cast(window)->SetValue(text_value) : dynamic_cast(window)->SetSelection(idx); break; } case coEnum:{ int id_value = static_cast*>(opt.default_value)->value; //!! dynamic_cast(window)->SetSelection(id_value); break; } } } void Choice::set_value(const std::string value) //! Redundant? { disable_change_event = true; auto idx=0; for (auto el : opt.enum_values) { if (el.compare(value) == 0) break; ++idx; } idx == opt.enum_values.size() ? dynamic_cast(window)->SetValue(value) : dynamic_cast(window)->SetSelection(idx); disable_change_event = false; } //! it's needed for _update_serial_ports() void Choice::set_values(const std::vector values) { disable_change_event = true; // # it looks that Clear() also clears the text field in recent wxWidgets versions, // # but we want to preserve it auto ww = dynamic_cast(window); auto value = ww->GetValue(); ww->Clear(); for (auto el : values) ww->Append(wxString(el)); ww->SetValue(value); disable_change_event = false; } } // GUI } // Slic3r