2018-11-26 13:41:58 +00:00
|
|
|
#include "GUI.hpp"
|
|
|
|
#include "GUI_App.hpp"
|
|
|
|
#include "I18N.hpp"
|
2017-12-13 13:45:10 +00:00
|
|
|
#include "Field.hpp"
|
2019-03-07 15:36:39 +00:00
|
|
|
#include "wxExtensions.hpp"
|
2020-05-26 15:42:57 +00:00
|
|
|
#include "Plater.hpp"
|
|
|
|
#include "MainFrame.hpp"
|
2020-12-07 18:47:37 +00:00
|
|
|
#include "format.hpp"
|
2017-12-13 13:45:10 +00:00
|
|
|
|
2018-11-26 13:41:58 +00:00
|
|
|
#include "libslic3r/PrintConfig.hpp"
|
|
|
|
|
2017-12-18 12:58:51 +00:00
|
|
|
#include <regex>
|
|
|
|
#include <wx/numformatter.h>
|
2018-02-01 08:29:07 +00:00
|
|
|
#include <wx/tooltip.h>
|
2020-05-12 07:46:23 +00:00
|
|
|
#include <wx/notebook.h>
|
2020-12-07 15:14:40 +00:00
|
|
|
#include <wx/tokenzr.h>
|
2018-01-27 13:21:16 +00:00
|
|
|
#include <boost/algorithm/string/predicate.hpp>
|
2020-10-23 17:17:10 +00:00
|
|
|
#include "OG_CustomCtrl.hpp"
|
2021-02-10 10:24:25 +00:00
|
|
|
#include "MsgDialog.hpp"
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2019-11-12 12:54:36 +00:00
|
|
|
#ifdef __WXOSX__
|
|
|
|
#define wxOSX true
|
|
|
|
#else
|
|
|
|
#define wxOSX false
|
|
|
|
#endif
|
|
|
|
|
2017-12-13 13:45:10 +00:00
|
|
|
namespace Slic3r { namespace GUI {
|
|
|
|
|
2018-11-14 11:33:48 +00:00
|
|
|
wxString double_to_string(double const value, const int max_precision /*= 4*/)
|
2018-10-31 11:56:08 +00:00
|
|
|
{
|
2019-05-10 09:44:21 +00:00
|
|
|
// Style_NoTrailingZeroes does not work on OSX. It also does not work correctly with some locales on Windows.
|
|
|
|
// return wxNumberFormatter::ToString(value, max_precision, wxNumberFormatter::Style_NoTrailingZeroes);
|
|
|
|
|
|
|
|
wxString s = wxNumberFormatter::ToString(value, max_precision, wxNumberFormatter::Style_None);
|
|
|
|
|
|
|
|
// The following code comes from wxNumberFormatter::RemoveTrailingZeroes(wxString& s)
|
|
|
|
// with the exception that here one sets the decimal separator explicitely to dot.
|
|
|
|
// If number is in scientific format, trailing zeroes belong to the exponent and cannot be removed.
|
|
|
|
if (s.find_first_of("eE") == wxString::npos) {
|
|
|
|
const size_t posDecSep = s.find(".");
|
|
|
|
// No decimal point => removing trailing zeroes irrelevant for integer number.
|
|
|
|
if (posDecSep != wxString::npos) {
|
|
|
|
// Find the last character to keep.
|
|
|
|
size_t posLastNonZero = s.find_last_not_of("0");
|
|
|
|
// If it's the decimal separator itself, don't keep it neither.
|
|
|
|
if (posLastNonZero == posDecSep)
|
|
|
|
-- posLastNonZero;
|
|
|
|
s.erase(posLastNonZero + 1);
|
|
|
|
// Remove sign from orphaned zero.
|
|
|
|
if (s.compare("-0") == 0)
|
|
|
|
s = "0";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
2018-10-31 11:56:08 +00:00
|
|
|
}
|
2018-02-19 08:15:15 +00:00
|
|
|
|
2020-12-07 15:14:40 +00:00
|
|
|
wxString get_thumbnails_string(const std::vector<Vec2d>& values)
|
|
|
|
{
|
|
|
|
wxString ret_str;
|
2020-12-12 07:34:43 +00:00
|
|
|
for (size_t i = 0; i < values.size(); ++ i) {
|
|
|
|
const Vec2d& el = values[i];
|
|
|
|
ret_str += wxString::Format((i == 0) ? "%ix%i" : ", %ix%i", int(el[0]), int(el[1]));
|
|
|
|
}
|
2020-12-07 15:14:40 +00:00
|
|
|
return ret_str;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-09-22 09:17:43 +00:00
|
|
|
Field::~Field()
|
|
|
|
{
|
|
|
|
if (m_on_kill_focus)
|
|
|
|
m_on_kill_focus = nullptr;
|
|
|
|
if (m_on_set_focus)
|
|
|
|
m_on_set_focus = nullptr;
|
|
|
|
if (m_on_change)
|
|
|
|
m_on_change = nullptr;
|
|
|
|
if (m_back_to_initial_value)
|
|
|
|
m_back_to_initial_value = nullptr;
|
|
|
|
if (m_back_to_sys_value)
|
|
|
|
m_back_to_sys_value = nullptr;
|
2020-10-13 10:17:39 +00:00
|
|
|
if (getWindow()) {
|
|
|
|
wxWindow* win = getWindow();
|
|
|
|
win->Destroy();
|
|
|
|
win = nullptr;
|
|
|
|
}
|
2020-09-22 09:17:43 +00:00
|
|
|
}
|
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
void Field::PostInitialize()
|
|
|
|
{
|
|
|
|
auto color = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW);
|
|
|
|
|
|
|
|
switch (m_opt.type)
|
2018-01-31 15:46:17 +00:00
|
|
|
{
|
2018-10-31 11:56:08 +00:00
|
|
|
case coPercents:
|
|
|
|
case coFloats:
|
2021-02-24 08:22:31 +00:00
|
|
|
case coStrings:
|
|
|
|
case coBools:
|
2018-10-31 11:56:08 +00:00
|
|
|
case coInts: {
|
|
|
|
auto tag_pos = m_opt_id.find("#");
|
|
|
|
if (tag_pos != std::string::npos)
|
|
|
|
m_opt_idx = stoi(m_opt_id.substr(tag_pos + 1, m_opt_id.size()));
|
|
|
|
break;
|
2018-03-06 11:34:20 +00:00
|
|
|
}
|
2018-10-31 11:56:08 +00:00
|
|
|
default:
|
|
|
|
break;
|
2018-03-16 16:25:11 +00:00
|
|
|
}
|
|
|
|
|
2019-04-17 19:35:53 +00:00
|
|
|
// initialize m_unit_value
|
|
|
|
m_em_unit = em_unit(m_parent);
|
2020-10-23 17:17:10 +00:00
|
|
|
parent_is_custom_ctrl = dynamic_cast<OG_CustomCtrl*>(m_parent) != nullptr;
|
2019-04-17 19:35:53 +00:00
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
BUILD();
|
2020-05-11 20:00:26 +00:00
|
|
|
|
|
|
|
// For the mode, when settings are in non-modal dialog, neither dialog nor tabpanel doesn't receive wxEVT_KEY_UP event, when some field is selected.
|
2021-02-24 08:22:31 +00:00
|
|
|
// So, like a workaround check wxEVT_KEY_UP event for the Filed and switch between tabs if Ctrl+(1-4) was pressed
|
2020-05-11 20:00:26 +00:00
|
|
|
if (getWindow())
|
|
|
|
getWindow()->Bind(wxEVT_KEY_UP, [](wxKeyEvent& evt) {
|
|
|
|
if ((evt.GetModifiers() & wxMOD_CONTROL) != 0) {
|
|
|
|
int tab_id = -1;
|
|
|
|
switch (evt.GetKeyCode()) {
|
|
|
|
case '1': { tab_id = 0; break; }
|
|
|
|
case '2': { tab_id = 1; break; }
|
|
|
|
case '3': { tab_id = 2; break; }
|
|
|
|
case '4': { tab_id = 3; break; }
|
2020-05-14 06:48:17 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
case 'f':
|
|
|
|
#else /* __APPLE__ */
|
|
|
|
case WXK_CONTROL_F:
|
|
|
|
#endif /* __APPLE__ */
|
|
|
|
case 'F': { wxGetApp().plater()->search(false); break; }
|
2020-05-11 20:00:26 +00:00
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
if (tab_id >= 0)
|
|
|
|
wxGetApp().mainframe->select_tab(tab_id);
|
|
|
|
if (tab_id > 0)
|
|
|
|
// tab panel should be focused for correct navigation between tabs
|
|
|
|
wxGetApp().tab_panel()->SetFocus();
|
|
|
|
}
|
2021-02-24 08:22:31 +00:00
|
|
|
|
2020-05-11 20:00:26 +00:00
|
|
|
evt.Skip();
|
|
|
|
});
|
2018-10-31 11:56:08 +00:00
|
|
|
}
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2020-02-18 10:09:28 +00:00
|
|
|
// Values of width to alignments of fields
|
2020-11-30 12:48:22 +00:00
|
|
|
int Field::def_width() { return 8; }
|
2020-07-06 16:49:06 +00:00
|
|
|
int Field::def_width_wider() { return 16; }
|
2020-02-18 10:09:28 +00:00
|
|
|
int Field::def_width_thinner() { return 4; }
|
|
|
|
|
2019-01-08 15:05:49 +00:00
|
|
|
void Field::on_kill_focus()
|
2018-10-31 11:56:08 +00:00
|
|
|
{
|
|
|
|
// call the registered function if it is available
|
2021-02-24 08:22:31 +00:00
|
|
|
if (m_on_kill_focus!=nullptr)
|
2018-12-11 12:34:37 +00:00
|
|
|
m_on_kill_focus(m_opt_id);
|
2018-10-31 11:56:08 +00:00
|
|
|
}
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2018-12-11 12:57:50 +00:00
|
|
|
void Field::on_set_focus(wxEvent& event)
|
|
|
|
{
|
|
|
|
// to allow the default behavior
|
|
|
|
event.Skip();
|
|
|
|
// call the registered function if it is available
|
2021-02-24 08:22:31 +00:00
|
|
|
if (m_on_set_focus!=nullptr)
|
2018-12-11 12:57:50 +00:00
|
|
|
m_on_set_focus(m_opt_id);
|
|
|
|
}
|
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
void Field::on_change_field()
|
|
|
|
{
|
2018-11-01 11:33:56 +00:00
|
|
|
// std::cerr << "calling Field::_on_change \n";
|
2018-10-31 11:56:08 +00:00
|
|
|
if (m_on_change != nullptr && !m_disable_change_event)
|
|
|
|
m_on_change(m_opt_id, get_value());
|
|
|
|
}
|
|
|
|
|
|
|
|
void Field::on_back_to_initial_value()
|
|
|
|
{
|
|
|
|
if (m_back_to_initial_value != nullptr && m_is_modified_value)
|
|
|
|
m_back_to_initial_value(m_opt_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Field::on_back_to_sys_value()
|
|
|
|
{
|
|
|
|
if (m_back_to_sys_value != nullptr && m_is_nonsys_value)
|
|
|
|
m_back_to_sys_value(m_opt_id);
|
|
|
|
}
|
2018-01-09 12:52:01 +00:00
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
wxString Field::get_tooltip_text(const wxString& default_string)
|
|
|
|
{
|
|
|
|
wxString tooltip_text("");
|
|
|
|
wxString tooltip = _(m_opt.tooltip);
|
2019-04-26 08:52:38 +00:00
|
|
|
edit_tooltip(tooltip);
|
2019-07-01 14:56:38 +00:00
|
|
|
|
|
|
|
std::string opt_id = m_opt_id;
|
|
|
|
auto hash_pos = opt_id.find("#");
|
|
|
|
if (hash_pos != std::string::npos) {
|
|
|
|
opt_id.replace(hash_pos, 1,"[");
|
|
|
|
opt_id += "]";
|
|
|
|
}
|
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
if (tooltip.length() > 0)
|
|
|
|
tooltip_text = tooltip + "\n" + _(L("default value")) + "\t: " +
|
2019-07-01 14:56:38 +00:00
|
|
|
(boost::iends_with(opt_id, "_gcode") ? "\n" : "") + default_string +
|
2021-02-24 08:22:31 +00:00
|
|
|
(boost::iends_with(opt_id, "_gcode") ? "" : "\n") +
|
2019-07-01 14:56:38 +00:00
|
|
|
_(L("parameter name")) + "\t: " + opt_id;
|
2018-10-31 11:56:08 +00:00
|
|
|
|
|
|
|
return tooltip_text;
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
bool Field::is_matched(const std::string& string, const 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);
|
|
|
|
}
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2019-07-29 12:00:22 +00:00
|
|
|
static wxString na_value() { return _(L("N/A")); }
|
|
|
|
|
2019-09-02 12:02:26 +00:00
|
|
|
void Field::get_value_by_opt_type(wxString& str, const bool check_value/* = true*/)
|
2018-10-31 11:56:08 +00:00
|
|
|
{
|
|
|
|
switch (m_opt.type) {
|
|
|
|
case coInt:
|
|
|
|
m_value = wxAtoi(str);
|
|
|
|
break;
|
|
|
|
case coPercent:
|
|
|
|
case coPercents:
|
|
|
|
case coFloats:
|
|
|
|
case coFloat:{
|
2021-02-24 08:22:31 +00:00
|
|
|
if (m_opt.type == coPercent && !str.IsEmpty() && str.Last() == '%')
|
2018-10-31 11:56:08 +00:00
|
|
|
str.RemoveLast();
|
2019-10-29 12:07:57 +00:00
|
|
|
else if (!str.IsEmpty() && str.Last() == '%')
|
|
|
|
{
|
|
|
|
if (!check_value) {
|
|
|
|
m_value.clear();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-11-02 14:28:43 +00:00
|
|
|
wxString label = m_opt.full_label.empty() ? _(m_opt.label) : _(m_opt.full_label);
|
2020-03-02 10:48:31 +00:00
|
|
|
show_error(m_parent, from_u8((boost::format(_utf8(L("%s doesn't support percentage"))) % label).str()));
|
2018-10-31 11:56:08 +00:00
|
|
|
set_value(double_to_string(m_opt.min), true);
|
|
|
|
m_value = double(m_opt.min);
|
2017-12-18 12:58:51 +00:00
|
|
|
break;
|
|
|
|
}
|
2018-10-31 11:56:08 +00:00
|
|
|
double val;
|
2019-01-29 09:21:21 +00:00
|
|
|
// Replace the first occurence of comma in decimal number.
|
|
|
|
str.Replace(",", ".", false);
|
2019-02-06 08:57:52 +00:00
|
|
|
if (str == ".")
|
|
|
|
val = 0.0;
|
|
|
|
else
|
|
|
|
{
|
2019-07-29 12:00:22 +00:00
|
|
|
if (m_opt.nullable && str == na_value())
|
|
|
|
val = ConfigOptionFloatsNullable::nil_value();
|
2019-10-29 12:07:57 +00:00
|
|
|
else if (!str.ToCDouble(&val))
|
2019-02-06 08:57:52 +00:00
|
|
|
{
|
2019-10-29 12:07:57 +00:00
|
|
|
if (!check_value) {
|
|
|
|
m_value.clear();
|
|
|
|
break;
|
|
|
|
}
|
2019-02-06 08:57:52 +00:00
|
|
|
show_error(m_parent, _(L("Invalid numeric input.")));
|
|
|
|
set_value(double_to_string(val), true);
|
|
|
|
}
|
2019-10-29 12:07:57 +00:00
|
|
|
if (m_opt.min > val || val > m_opt.max)
|
2019-02-06 08:57:52 +00:00
|
|
|
{
|
2019-10-29 12:07:57 +00:00
|
|
|
if (!check_value) {
|
|
|
|
m_value.clear();
|
|
|
|
break;
|
|
|
|
}
|
2020-12-15 20:21:09 +00:00
|
|
|
if (m_opt_id == "extrusion_multiplier") {
|
2020-12-16 12:07:27 +00:00
|
|
|
if (m_value.empty() || boost::any_cast<double>(m_value) != val) {
|
|
|
|
wxString msg_text = format_wxstr(_L("Input value is out of range\n"
|
2020-12-16 16:28:42 +00:00
|
|
|
"Are you sure that %s is a correct value and that you want to continue?"), str);
|
2021-02-10 10:24:25 +00:00
|
|
|
// wxMessageDialog dialog(m_parent, msg_text, _L("Parameter validation") + ": " + m_opt_id, wxICON_WARNING | wxYES | wxNO);
|
|
|
|
WarningDialog dialog(m_parent, msg_text, _L("Parameter validation") + ": " + m_opt_id, wxYES | wxNO);
|
2020-12-16 19:04:14 +00:00
|
|
|
if (dialog.ShowModal() == wxID_NO) {
|
|
|
|
if (m_value.empty()) {
|
|
|
|
if (m_opt.min > val) val = m_opt.min;
|
|
|
|
if (val > m_opt.max) val = m_opt.max;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
val = boost::any_cast<double>(m_value);
|
|
|
|
set_value(double_to_string(val), true);
|
|
|
|
}
|
2020-12-16 12:07:27 +00:00
|
|
|
}
|
2020-12-15 20:21:09 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
show_error(m_parent, _L("Input value is out of range"));
|
2020-12-16 19:04:14 +00:00
|
|
|
if (m_opt.min > val) val = m_opt.min;
|
|
|
|
if (val > m_opt.max) val = m_opt.max;
|
|
|
|
set_value(double_to_string(val), true);
|
2020-12-15 20:21:09 +00:00
|
|
|
}
|
2019-02-06 08:57:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
m_value = val;
|
2018-10-31 11:56:08 +00:00
|
|
|
break; }
|
|
|
|
case coString:
|
|
|
|
case coStrings:
|
2018-12-17 17:07:31 +00:00
|
|
|
case coFloatOrPercent: {
|
|
|
|
if (m_opt.type == coFloatOrPercent && !str.IsEmpty() && str.Last() != '%')
|
|
|
|
{
|
2019-09-06 11:33:08 +00:00
|
|
|
double val = 0.;
|
2019-01-29 09:21:21 +00:00
|
|
|
// Replace the first occurence of comma in decimal number.
|
|
|
|
str.Replace(",", ".", false);
|
2020-11-28 00:17:01 +00:00
|
|
|
|
|
|
|
// remove space and "mm" substring, if any exists
|
|
|
|
str.Replace(" ", "", true);
|
|
|
|
str.Replace("m", "", true);
|
|
|
|
|
2019-10-29 12:07:57 +00:00
|
|
|
if (!str.ToCDouble(&val))
|
2018-12-17 17:07:31 +00:00
|
|
|
{
|
2019-10-29 12:07:57 +00:00
|
|
|
if (!check_value) {
|
|
|
|
m_value.clear();
|
|
|
|
break;
|
|
|
|
}
|
2019-01-29 09:21:21 +00:00
|
|
|
show_error(m_parent, _(L("Invalid numeric input.")));
|
|
|
|
set_value(double_to_string(val), true);
|
2018-12-17 17:07:31 +00:00
|
|
|
}
|
2019-10-29 12:07:57 +00:00
|
|
|
else if (((m_opt.sidetext.rfind("mm/s") != std::string::npos && val > m_opt.max) ||
|
2019-09-06 11:33:08 +00:00
|
|
|
(m_opt.sidetext.rfind("mm ") != std::string::npos && val > 1)) &&
|
2019-07-29 16:02:48 +00:00
|
|
|
(m_value.empty() || std::string(str.ToUTF8().data()) != boost::any_cast<std::string>(m_value)))
|
2018-12-17 17:07:31 +00:00
|
|
|
{
|
2019-10-29 12:07:57 +00:00
|
|
|
if (!check_value) {
|
|
|
|
m_value.clear();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-11-28 00:17:01 +00:00
|
|
|
bool infill_anchors = m_opt.opt_key == "infill_anchor" || m_opt.opt_key == "infill_anchor_max";
|
|
|
|
|
2019-07-29 16:02:48 +00:00
|
|
|
const std::string sidetext = m_opt.sidetext.rfind("mm/s") != std::string::npos ? "mm/s" : "mm";
|
|
|
|
const wxString stVal = double_to_string(val, 2);
|
2020-03-02 10:48:31 +00:00
|
|
|
const wxString msg_text = from_u8((boost::format(_utf8(L("Do you mean %s%% instead of %s %s?\n"
|
2019-07-29 16:02:48 +00:00
|
|
|
"Select YES if you want to change this value to %s%%, \n"
|
2020-03-02 10:48:31 +00:00
|
|
|
"or NO if you are sure that %s %s is a correct value."))) % stVal % stVal % sidetext % stVal % stVal % sidetext).str());
|
2021-02-10 10:24:25 +00:00
|
|
|
// wxMessageDialog dialog(m_parent, msg_text, _(L("Parameter validation")) + ": " + m_opt_id , wxICON_WARNING | wxYES | wxNO);
|
|
|
|
WarningDialog dialog(m_parent, msg_text, _L("Parameter validation") + ": " + m_opt_id, wxYES | wxNO);
|
2020-11-28 00:17:01 +00:00
|
|
|
if ((!infill_anchors || val > 100) && dialog.ShowModal() == wxID_YES) {
|
2020-03-02 10:48:31 +00:00
|
|
|
set_value(from_u8((boost::format("%s%%") % stVal).str()), false/*true*/);
|
2018-12-17 17:07:31 +00:00
|
|
|
str += "%%";
|
|
|
|
}
|
2020-02-18 10:09:28 +00:00
|
|
|
else
|
|
|
|
set_value(stVal, false); // it's no needed but can be helpful, when inputted value contained "," instead of "."
|
2018-12-17 17:07:31 +00:00
|
|
|
}
|
|
|
|
}
|
2021-02-24 08:22:31 +00:00
|
|
|
|
2019-01-07 08:37:51 +00:00
|
|
|
m_value = std::string(str.ToUTF8().data());
|
2018-12-17 17:07:31 +00:00
|
|
|
break; }
|
2020-12-07 15:14:40 +00:00
|
|
|
|
|
|
|
case coPoints: {
|
|
|
|
std::vector<Vec2d> out_values;
|
|
|
|
str.Replace(" ", wxEmptyString, true);
|
|
|
|
if (!str.IsEmpty()) {
|
|
|
|
bool invalid_val = false;
|
2020-12-08 12:29:23 +00:00
|
|
|
bool out_of_range_val = false;
|
2020-12-07 15:14:40 +00:00
|
|
|
wxStringTokenizer thumbnails(str, ",");
|
|
|
|
while (thumbnails.HasMoreTokens()) {
|
|
|
|
wxString token = thumbnails.GetNextToken();
|
|
|
|
double x, y;
|
|
|
|
wxStringTokenizer thumbnail(token, "x");
|
|
|
|
if (thumbnail.HasMoreTokens()) {
|
|
|
|
wxString x_str = thumbnail.GetNextToken();
|
|
|
|
if (x_str.ToDouble(&x) && thumbnail.HasMoreTokens()) {
|
|
|
|
wxString y_str = thumbnail.GetNextToken();
|
|
|
|
if (y_str.ToDouble(&y) && !thumbnail.HasMoreTokens()) {
|
2020-12-08 12:29:23 +00:00
|
|
|
if (0 < x && x < 1000 && 0 < y && y < 1000) {
|
|
|
|
out_values.push_back(Vec2d(x, y));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
out_of_range_val = true;
|
|
|
|
break;
|
2020-12-07 15:14:40 +00:00
|
|
|
}
|
2021-02-24 08:22:31 +00:00
|
|
|
}
|
|
|
|
}
|
2020-12-07 15:14:40 +00:00
|
|
|
invalid_val = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2020-12-08 12:29:23 +00:00
|
|
|
if (out_of_range_val) {
|
|
|
|
wxString text_value;
|
|
|
|
if (!m_value.empty())
|
|
|
|
text_value = get_thumbnails_string(boost::any_cast<std::vector<Vec2d>>(m_value));
|
|
|
|
set_value(text_value, true);
|
2021-02-10 10:24:25 +00:00
|
|
|
show_error(m_parent, _L("Input value is out of range"));
|
2020-12-08 12:29:23 +00:00
|
|
|
}
|
|
|
|
else if (invalid_val) {
|
2020-12-07 15:14:40 +00:00
|
|
|
wxString text_value;
|
|
|
|
if (!m_value.empty())
|
|
|
|
text_value = get_thumbnails_string(boost::any_cast<std::vector<Vec2d>>(m_value));
|
|
|
|
set_value(text_value, true);
|
2020-12-12 07:34:43 +00:00
|
|
|
show_error(m_parent, format_wxstr(_L("Invalid input format. Expected vector of dimensions in the following format: \"%1%\""),"XxY, XxY, ..." ));
|
2020-12-07 15:14:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_value = out_values;
|
|
|
|
break; }
|
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2020-11-02 14:28:43 +00:00
|
|
|
void Field::msw_rescale()
|
2020-02-12 11:08:43 +00:00
|
|
|
{
|
|
|
|
// update em_unit value
|
|
|
|
m_em_unit = em_unit(m_parent);
|
|
|
|
}
|
|
|
|
|
2020-05-22 13:23:05 +00:00
|
|
|
void Field::sys_color_changed()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2018-12-18 12:22:22 +00:00
|
|
|
template<class T>
|
|
|
|
bool is_defined_input_value(wxWindow* win, const ConfigOptionType& type)
|
2018-11-21 14:02:22 +00:00
|
|
|
{
|
2020-09-22 09:17:43 +00:00
|
|
|
if (!win || (static_cast<T*>(win)->GetValue().empty() && type != coString && type != coStrings))
|
2018-11-21 14:02:22 +00:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
void TextCtrl::BUILD() {
|
2020-02-12 11:08:43 +00:00
|
|
|
auto size = wxSize(def_width()*m_em_unit, wxDefaultCoord);
|
2019-04-17 19:35:53 +00:00
|
|
|
if (m_opt.height >= 0) size.SetHeight(m_opt.height*m_em_unit);
|
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width*m_em_unit);
|
2017-12-13 13:45:10 +00:00
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
wxString text_value = wxString("");
|
2018-10-31 11:56:08 +00:00
|
|
|
|
|
|
|
switch (m_opt.type) {
|
|
|
|
case coFloatOrPercent:
|
|
|
|
{
|
|
|
|
text_value = double_to_string(m_opt.default_value->getFloat());
|
2019-05-03 16:30:58 +00:00
|
|
|
if (m_opt.get_default_value<ConfigOptionFloatOrPercent>()->percent)
|
2018-10-31 11:56:08 +00:00
|
|
|
text_value += "%";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case coPercent:
|
|
|
|
{
|
|
|
|
text_value = wxString::Format(_T("%i"), int(m_opt.default_value->getFloat()));
|
|
|
|
text_value += "%";
|
|
|
|
break;
|
2021-02-24 08:22:31 +00:00
|
|
|
}
|
2018-10-31 11:56:08 +00:00
|
|
|
case coPercents:
|
|
|
|
case coFloats:
|
|
|
|
case coFloat:
|
|
|
|
{
|
|
|
|
double val = m_opt.type == coFloats ?
|
2019-05-03 16:30:58 +00:00
|
|
|
m_opt.get_default_value<ConfigOptionFloats>()->get_at(m_opt_idx) :
|
2021-02-24 08:22:31 +00:00
|
|
|
m_opt.type == coFloat ?
|
2018-10-31 11:56:08 +00:00
|
|
|
m_opt.default_value->getFloat() :
|
2019-05-03 16:30:58 +00:00
|
|
|
m_opt.get_default_value<ConfigOptionPercents>()->get_at(m_opt_idx);
|
2018-10-31 11:56:08 +00:00
|
|
|
text_value = double_to_string(val);
|
2019-07-28 20:00:39 +00:00
|
|
|
m_last_meaningful_value = text_value;
|
2018-10-31 11:56:08 +00:00
|
|
|
break;
|
|
|
|
}
|
2021-02-24 08:22:31 +00:00
|
|
|
case coString:
|
2019-05-03 16:30:58 +00:00
|
|
|
text_value = m_opt.get_default_value<ConfigOptionString>()->value;
|
2018-10-31 11:56:08 +00:00
|
|
|
break;
|
|
|
|
case coStrings:
|
|
|
|
{
|
2019-05-03 16:30:58 +00:00
|
|
|
const ConfigOptionStrings *vec = m_opt.get_default_value<ConfigOptionStrings>();
|
2018-10-31 11:56:08 +00:00
|
|
|
if (vec == nullptr || vec->empty()) break; //for the case of empty default value
|
|
|
|
text_value = vec->get_at(m_opt_idx);
|
|
|
|
break;
|
|
|
|
}
|
2020-12-07 15:14:40 +00:00
|
|
|
case coPoints:
|
|
|
|
text_value = get_thumbnails_string(m_opt.get_default_value<ConfigOptionPoints>()->values);
|
|
|
|
break;
|
2018-10-31 11:56:08 +00:00
|
|
|
default:
|
2021-02-24 08:22:31 +00:00
|
|
|
break;
|
2018-10-31 11:56:08 +00:00
|
|
|
}
|
|
|
|
|
2019-01-08 15:05:49 +00:00
|
|
|
const long style = m_opt.multiline ? wxTE_MULTILINE : wxTE_PROCESS_ENTER/*0*/;
|
2018-11-14 11:33:48 +00:00
|
|
|
auto temp = new wxTextCtrl(m_parent, wxID_ANY, text_value, wxDefaultPosition, size, style);
|
2020-10-23 17:17:10 +00:00
|
|
|
if (parent_is_custom_ctrl && m_opt.height < 0)
|
|
|
|
opt_height = (double)temp->GetSize().GetHeight()/m_em_unit;
|
2020-09-13 00:19:36 +00:00
|
|
|
temp->SetFont(m_opt.is_code ?
|
|
|
|
Slic3r::GUI::wxGetApp().code_font():
|
|
|
|
Slic3r::GUI::wxGetApp().normal_font());
|
2019-03-18 19:54:01 +00:00
|
|
|
|
2019-11-12 12:54:36 +00:00
|
|
|
if (! m_opt.multiline && !wxOSX)
|
2019-03-18 19:54:01 +00:00
|
|
|
// Only disable background refresh for single line input fields, as they are completely painted over by the edit control.
|
|
|
|
// This does not apply to the multi-line edit field, where the last line and a narrow frame around the text is not cleared.
|
|
|
|
temp->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2019-01-11 15:32:16 +00:00
|
|
|
#ifdef __WXOSX__
|
|
|
|
temp->OSXDisableAllSmartSubstitutions();
|
|
|
|
#endif // __WXOSX__
|
2018-10-31 11:56:08 +00:00
|
|
|
|
|
|
|
temp->SetToolTip(get_tooltip_text(text_value));
|
2018-12-11 12:57:50 +00:00
|
|
|
|
2019-01-08 15:05:49 +00:00
|
|
|
if (style == wxTE_PROCESS_ENTER) {
|
|
|
|
temp->Bind(wxEVT_TEXT_ENTER, ([this, temp](wxEvent& e)
|
|
|
|
{
|
|
|
|
#if !defined(__WXGTK__)
|
|
|
|
e.Skip();
|
|
|
|
temp->GetToolTip()->Enable(true);
|
|
|
|
#endif // __WXGTK__
|
|
|
|
bEnterPressed = true;
|
2019-05-06 13:24:50 +00:00
|
|
|
propagate_value();
|
2019-01-08 15:05:49 +00:00
|
|
|
}), temp->GetId());
|
|
|
|
}
|
|
|
|
|
2018-12-11 12:57:50 +00:00
|
|
|
temp->Bind(wxEVT_SET_FOCUS, ([this](wxEvent& e) { on_set_focus(e); }), temp->GetId());
|
2021-02-24 08:22:31 +00:00
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
temp->Bind(wxEVT_LEFT_DOWN, ([temp](wxEvent& event)
|
|
|
|
{
|
|
|
|
//! to allow the default handling
|
|
|
|
event.Skip();
|
|
|
|
//! eliminating the g-code pop up text description
|
|
|
|
bool flag = false;
|
2018-05-11 12:41:21 +00:00
|
|
|
#ifdef __WXGTK__
|
2018-10-31 11:56:08 +00:00
|
|
|
// I have no idea why, but on GTK flag works in other way
|
|
|
|
flag = true;
|
2018-05-11 12:41:21 +00:00
|
|
|
#endif // __WXGTK__
|
2018-10-31 11:56:08 +00:00
|
|
|
temp->GetToolTip()->Enable(flag);
|
|
|
|
}), temp->GetId());
|
2018-01-31 13:59:44 +00:00
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
temp->Bind(wxEVT_KILL_FOCUS, ([this, temp](wxEvent& e)
|
|
|
|
{
|
2018-11-21 14:02:22 +00:00
|
|
|
e.Skip();
|
2020-02-18 10:09:28 +00:00
|
|
|
#ifdef __WXOSX__
|
2021-02-24 08:22:31 +00:00
|
|
|
// OSX issue: For some unknown reason wxEVT_KILL_FOCUS is emitted twice in a row in some cases
|
2020-03-10 14:19:34 +00:00
|
|
|
// (like when information dialog is shown during an update of the option value)
|
2020-02-18 10:09:28 +00:00
|
|
|
// Thus, suppress its second call
|
2020-03-10 14:19:34 +00:00
|
|
|
if (bKilledFocus)
|
2020-02-18 10:09:28 +00:00
|
|
|
return;
|
|
|
|
bKilledFocus = true;
|
|
|
|
#endif // __WXOSX__
|
|
|
|
|
2019-01-08 15:05:49 +00:00
|
|
|
#if !defined(__WXGTK__)
|
2018-10-31 11:56:08 +00:00
|
|
|
temp->GetToolTip()->Enable(true);
|
2018-05-11 12:41:21 +00:00
|
|
|
#endif // __WXGTK__
|
2020-02-18 10:09:28 +00:00
|
|
|
if (bEnterPressed)
|
2019-01-08 15:05:49 +00:00
|
|
|
bEnterPressed = false;
|
2020-02-18 10:09:28 +00:00
|
|
|
else
|
|
|
|
propagate_value();
|
2020-03-10 14:19:34 +00:00
|
|
|
#ifdef __WXOSX__
|
|
|
|
// After processing of KILL_FOCUS event we should to invalidate a bKilledFocus flag
|
|
|
|
bKilledFocus = false;
|
|
|
|
#endif // __WXOSX__
|
2018-11-21 14:02:22 +00:00
|
|
|
}), temp->GetId());
|
2020-05-11 20:00:26 +00:00
|
|
|
/*
|
2018-10-31 11:56:08 +00:00
|
|
|
// select all text using Ctrl+A
|
|
|
|
temp->Bind(wxEVT_CHAR, ([temp](wxKeyEvent& event)
|
|
|
|
{
|
|
|
|
if (wxGetKeyState(wxKeyCode('A')) && wxGetKeyState(WXK_CONTROL))
|
|
|
|
temp->SetSelection(-1, -1); //select all
|
|
|
|
event.Skip();
|
|
|
|
}));
|
2020-05-11 20:00:26 +00:00
|
|
|
*/
|
2018-10-31 11:56:08 +00:00
|
|
|
// recast as a wxWindow to fit the calling convention
|
|
|
|
window = dynamic_cast<wxWindow*>(temp);
|
2021-02-24 08:22:31 +00:00
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
2019-07-30 12:16:07 +00:00
|
|
|
bool TextCtrl::value_was_changed()
|
|
|
|
{
|
|
|
|
if (m_value.empty())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
boost::any val = m_value;
|
|
|
|
wxString ret_str = static_cast<wxTextCtrl*>(window)->GetValue();
|
|
|
|
// update m_value!
|
2019-10-29 12:07:57 +00:00
|
|
|
// ret_str might be changed inside get_value_by_opt_type
|
2019-07-30 12:16:07 +00:00
|
|
|
get_value_by_opt_type(ret_str);
|
|
|
|
|
|
|
|
switch (m_opt.type) {
|
|
|
|
case coInt:
|
|
|
|
return boost::any_cast<int>(m_value) != boost::any_cast<int>(val);
|
|
|
|
case coPercent:
|
|
|
|
case coPercents:
|
|
|
|
case coFloats:
|
|
|
|
case coFloat: {
|
2021-02-24 08:22:31 +00:00
|
|
|
if (m_opt.nullable && std::isnan(boost::any_cast<double>(m_value)) &&
|
2019-07-30 12:16:07 +00:00
|
|
|
std::isnan(boost::any_cast<double>(val)))
|
|
|
|
return false;
|
|
|
|
return boost::any_cast<double>(m_value) != boost::any_cast<double>(val);
|
|
|
|
}
|
|
|
|
case coString:
|
|
|
|
case coStrings:
|
|
|
|
case coFloatOrPercent:
|
|
|
|
return boost::any_cast<std::string>(m_value) != boost::any_cast<std::string>(val);
|
|
|
|
default:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-08 15:05:49 +00:00
|
|
|
void TextCtrl::propagate_value()
|
|
|
|
{
|
2020-02-18 10:09:28 +00:00
|
|
|
if (!is_defined_input_value<wxTextCtrl>(window, m_opt.type) )
|
|
|
|
// on_kill_focus() cause a call of OptionsGroup::reload_config(),
|
|
|
|
// Thus, do it only when it's really needed (when undefined value was input)
|
2019-01-08 15:05:49 +00:00
|
|
|
on_kill_focus();
|
2020-12-16 12:07:27 +00:00
|
|
|
else if (value_was_changed())
|
|
|
|
on_change_field();
|
2019-01-08 15:05:49 +00:00
|
|
|
}
|
|
|
|
|
2019-07-29 12:00:22 +00:00
|
|
|
void TextCtrl::set_value(const boost::any& value, bool change_event/* = false*/) {
|
|
|
|
m_disable_change_event = !change_event;
|
|
|
|
if (m_opt.nullable) {
|
|
|
|
const bool m_is_na_val = boost::any_cast<wxString>(value) == na_value();
|
|
|
|
if (!m_is_na_val)
|
|
|
|
m_last_meaningful_value = value;
|
|
|
|
dynamic_cast<wxTextCtrl*>(window)->SetValue(m_is_na_val ? na_value() : boost::any_cast<wxString>(value));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
dynamic_cast<wxTextCtrl*>(window)->SetValue(boost::any_cast<wxString>(value));
|
|
|
|
m_disable_change_event = false;
|
2019-08-12 13:21:06 +00:00
|
|
|
|
|
|
|
if (!change_event) {
|
|
|
|
wxString ret_str = static_cast<wxTextCtrl*>(window)->GetValue();
|
2021-02-24 08:22:31 +00:00
|
|
|
/* Update m_value to correct work of next value_was_changed().
|
|
|
|
* But after checking of entered value, don't fix the "incorrect" value and don't show a warning message,
|
|
|
|
* just clear m_value in this case.
|
2019-10-29 12:07:57 +00:00
|
|
|
*/
|
2019-09-02 12:02:26 +00:00
|
|
|
get_value_by_opt_type(ret_str, false);
|
2019-08-12 13:21:06 +00:00
|
|
|
}
|
2019-07-29 12:00:22 +00:00
|
|
|
}
|
|
|
|
|
2019-07-28 20:00:39 +00:00
|
|
|
void TextCtrl::set_last_meaningful_value()
|
|
|
|
{
|
|
|
|
dynamic_cast<wxTextCtrl*>(window)->SetValue(boost::any_cast<wxString>(m_last_meaningful_value));
|
|
|
|
propagate_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextCtrl::set_na_value()
|
|
|
|
{
|
2019-07-29 12:00:22 +00:00
|
|
|
dynamic_cast<wxTextCtrl*>(window)->SetValue(na_value());
|
2019-07-28 20:00:39 +00:00
|
|
|
propagate_value();
|
|
|
|
}
|
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
boost::any& TextCtrl::get_value()
|
|
|
|
{
|
|
|
|
wxString ret_str = static_cast<wxTextCtrl*>(window)->GetValue();
|
2019-08-12 13:21:06 +00:00
|
|
|
// update m_value
|
2018-10-31 11:56:08 +00:00
|
|
|
get_value_by_opt_type(ret_str);
|
2018-01-09 12:52:01 +00:00
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
return m_value;
|
|
|
|
}
|
2017-12-13 13:45:10 +00:00
|
|
|
|
2020-11-02 14:28:43 +00:00
|
|
|
void TextCtrl::msw_rescale()
|
2019-04-13 21:46:52 +00:00
|
|
|
{
|
2020-11-02 14:28:43 +00:00
|
|
|
Field::msw_rescale();
|
2020-02-12 11:08:43 +00:00
|
|
|
auto size = wxSize(def_width() * m_em_unit, wxDefaultCoord);
|
2020-10-23 17:17:10 +00:00
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
if (m_opt.height >= 0)
|
2020-10-23 17:17:10 +00:00
|
|
|
size.SetHeight(m_opt.height*m_em_unit);
|
|
|
|
else if (parent_is_custom_ctrl && opt_height > 0)
|
|
|
|
size.SetHeight(lround(opt_height*m_em_unit));
|
2019-04-17 19:35:53 +00:00
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width*m_em_unit);
|
2019-04-13 21:46:52 +00:00
|
|
|
|
|
|
|
if (size != wxDefaultSize)
|
|
|
|
{
|
|
|
|
wxTextCtrl* field = dynamic_cast<wxTextCtrl*>(window);
|
2020-10-23 17:17:10 +00:00
|
|
|
if (parent_is_custom_ctrl)
|
|
|
|
field->SetSize(size);
|
|
|
|
else
|
|
|
|
field->SetMinSize(size);
|
2019-04-13 21:46:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
void TextCtrl::enable() { dynamic_cast<wxTextCtrl*>(window)->Enable(); dynamic_cast<wxTextCtrl*>(window)->SetEditable(true); }
|
|
|
|
void TextCtrl::disable() { dynamic_cast<wxTextCtrl*>(window)->Disable(); dynamic_cast<wxTextCtrl*>(window)->SetEditable(false); }
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2018-09-05 07:47:36 +00:00
|
|
|
#ifdef __WXGTK__
|
2018-10-31 11:56:08 +00:00
|
|
|
void TextCtrl::change_field_value(wxEvent& event)
|
|
|
|
{
|
2019-11-19 11:44:55 +00:00
|
|
|
if ((bChangedValueEvent = (event.GetEventType()==wxEVT_KEY_UP)))
|
2018-10-31 11:56:08 +00:00
|
|
|
on_change_field();
|
|
|
|
event.Skip();
|
|
|
|
};
|
2018-09-05 07:47:36 +00:00
|
|
|
#endif //__WXGTK__
|
|
|
|
|
2017-12-18 12:58:51 +00:00
|
|
|
void CheckBox::BUILD() {
|
|
|
|
auto size = wxSize(wxDefaultSize);
|
2019-04-25 13:06:44 +00:00
|
|
|
if (m_opt.height >= 0) size.SetHeight(m_opt.height*m_em_unit);
|
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width*m_em_unit);
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
bool check_value = m_opt.type == coBool ?
|
|
|
|
m_opt.default_value->getBool() : m_opt.type == coBools ?
|
|
|
|
m_opt.get_default_value<ConfigOptionBools>()->get_at(m_opt_idx) :
|
2019-05-03 16:30:58 +00:00
|
|
|
false;
|
2017-12-22 10:50:28 +00:00
|
|
|
|
2019-07-28 20:00:39 +00:00
|
|
|
m_last_meaningful_value = static_cast<unsigned char>(check_value);
|
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
// Set Label as a string of at least one space simbol to correct system scaling of a CheckBox
|
|
|
|
auto temp = new wxCheckBox(m_parent, wxID_ANY, wxString(" "), wxDefaultPosition, size);
|
2019-03-18 19:54:01 +00:00
|
|
|
temp->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-11-12 12:54:36 +00:00
|
|
|
if (!wxOSX) temp->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2017-12-22 10:50:28 +00:00
|
|
|
temp->SetValue(check_value);
|
2018-01-05 14:11:33 +00:00
|
|
|
if (m_opt.readonly) temp->Disable();
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2019-07-28 20:00:39 +00:00
|
|
|
temp->Bind(wxEVT_CHECKBOX, ([this](wxCommandEvent e) {
|
|
|
|
m_is_na_val = false;
|
|
|
|
on_change_field();
|
|
|
|
}), temp->GetId());
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
temp->SetToolTip(get_tooltip_text(check_value ? "true" : "false"));
|
2017-12-18 12:58:51 +00:00
|
|
|
|
|
|
|
// recast as a wxWindow to fit the calling convention
|
|
|
|
window = dynamic_cast<wxWindow*>(temp);
|
|
|
|
}
|
|
|
|
|
2019-07-28 20:00:39 +00:00
|
|
|
void CheckBox::set_value(const boost::any& value, bool change_event)
|
|
|
|
{
|
|
|
|
m_disable_change_event = !change_event;
|
|
|
|
if (m_opt.nullable) {
|
|
|
|
m_is_na_val = boost::any_cast<unsigned char>(value) == ConfigOptionBoolsNullable::nil_value();
|
|
|
|
if (!m_is_na_val)
|
|
|
|
m_last_meaningful_value = value;
|
|
|
|
dynamic_cast<wxCheckBox*>(window)->SetValue(m_is_na_val ? false : boost::any_cast<unsigned char>(value) != 0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
dynamic_cast<wxCheckBox*>(window)->SetValue(boost::any_cast<bool>(value));
|
|
|
|
m_disable_change_event = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheckBox::set_last_meaningful_value()
|
|
|
|
{
|
|
|
|
if (m_opt.nullable) {
|
|
|
|
m_is_na_val = false;
|
|
|
|
dynamic_cast<wxCheckBox*>(window)->SetValue(boost::any_cast<unsigned char>(m_last_meaningful_value) != 0);
|
|
|
|
on_change_field();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CheckBox::set_na_value()
|
|
|
|
{
|
|
|
|
if (m_opt.nullable) {
|
|
|
|
m_is_na_val = true;
|
|
|
|
dynamic_cast<wxCheckBox*>(window)->SetValue(false);
|
|
|
|
on_change_field();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-24 08:33:11 +00:00
|
|
|
boost::any& CheckBox::get_value()
|
2018-03-09 16:17:51 +00:00
|
|
|
{
|
2018-04-24 08:33:11 +00:00
|
|
|
// boost::any m_value;
|
2018-03-09 16:17:51 +00:00
|
|
|
bool value = dynamic_cast<wxCheckBox*>(window)->GetValue();
|
|
|
|
if (m_opt.type == coBool)
|
2018-04-24 08:33:11 +00:00
|
|
|
m_value = static_cast<bool>(value);
|
2018-03-09 16:17:51 +00:00
|
|
|
else
|
2019-07-28 20:00:39 +00:00
|
|
|
m_value = m_is_na_val ? ConfigOptionBoolsNullable::nil_value() : static_cast<unsigned char>(value);
|
2018-04-24 08:33:11 +00:00
|
|
|
return m_value;
|
2018-03-09 16:17:51 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 14:28:43 +00:00
|
|
|
void CheckBox::msw_rescale()
|
2019-04-25 13:06:44 +00:00
|
|
|
{
|
|
|
|
Field::msw_rescale();
|
|
|
|
|
|
|
|
wxCheckBox* field = dynamic_cast<wxCheckBox*>(window);
|
|
|
|
field->SetMinSize(wxSize(-1, int(1.5f*field->GetFont().GetPixelSize().y +0.5f)));
|
|
|
|
}
|
|
|
|
|
2017-12-18 12:58:51 +00:00
|
|
|
|
|
|
|
void SpinCtrl::BUILD() {
|
2020-02-12 11:08:43 +00:00
|
|
|
auto size = wxSize(def_width() * m_em_unit, wxDefaultCoord);
|
2019-04-17 19:35:53 +00:00
|
|
|
if (m_opt.height >= 0) size.SetHeight(m_opt.height*m_em_unit);
|
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width*m_em_unit);
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2017-12-22 10:50:28 +00:00
|
|
|
wxString text_value = wxString("");
|
|
|
|
int default_value = 0;
|
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
switch (m_opt.type) {
|
2017-12-22 10:50:28 +00:00
|
|
|
case coInt:
|
2018-01-05 14:11:33 +00:00
|
|
|
default_value = m_opt.default_value->getInt();
|
2017-12-22 10:50:28 +00:00
|
|
|
text_value = wxString::Format(_T("%i"), default_value);
|
|
|
|
break;
|
|
|
|
case coInts:
|
|
|
|
{
|
2019-05-03 16:30:58 +00:00
|
|
|
const ConfigOptionInts *vec = m_opt.get_default_value<ConfigOptionInts>();
|
2017-12-22 10:50:28 +00:00
|
|
|
if (vec == nullptr || vec->empty()) break;
|
|
|
|
for (size_t id = 0; id < vec->size(); ++id)
|
|
|
|
{
|
|
|
|
default_value = vec->get_at(id);
|
|
|
|
text_value += wxString::Format(_T("%i"), default_value);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
const int min_val = m_opt.min == INT_MIN
|
2019-08-19 10:58:59 +00:00
|
|
|
#ifdef __WXOSX__
|
2021-02-24 08:22:31 +00:00
|
|
|
// We will forcibly set the input value for SpinControl, since the value
|
2019-08-19 10:58:59 +00:00
|
|
|
// inserted from the keyboard is not updated under OSX.
|
|
|
|
// So, we can't set min control value bigger then 0.
|
2021-02-24 08:22:31 +00:00
|
|
|
// Otherwise, it couldn't be possible to input from keyboard value
|
2019-08-19 10:58:59 +00:00
|
|
|
// less then min_val.
|
2021-02-24 08:22:31 +00:00
|
|
|
|| m_opt.min > 0
|
2019-08-19 10:58:59 +00:00
|
|
|
#endif
|
|
|
|
? 0 : m_opt.min;
|
2018-03-08 15:33:38 +00:00
|
|
|
const int max_val = m_opt.max < 2147483647 ? m_opt.max : 2147483647;
|
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
auto temp = new wxSpinCtrl(m_parent, wxID_ANY, text_value, wxDefaultPosition, size,
|
2019-01-08 15:05:49 +00:00
|
|
|
0|wxTE_PROCESS_ENTER, min_val, max_val, default_value);
|
2020-11-12 20:43:21 +00:00
|
|
|
#ifdef __WXGTK3__
|
|
|
|
wxSize best_sz = temp->GetBestSize();
|
|
|
|
if (best_sz.x > size.x)
|
|
|
|
temp->SetSize(wxSize(size.x + 2 * best_sz.y, best_sz.y));
|
|
|
|
#endif //__WXGTK3__
|
2019-03-18 19:54:01 +00:00
|
|
|
temp->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-11-12 12:54:36 +00:00
|
|
|
if (!wxOSX) temp->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2020-10-23 17:17:10 +00:00
|
|
|
if (m_opt.height < 0 && parent_is_custom_ctrl)
|
|
|
|
opt_height = (double)temp->GetSize().GetHeight() / m_em_unit;
|
|
|
|
|
2019-05-20 11:32:36 +00:00
|
|
|
// XXX: On OS X the wxSpinCtrl widget is made up of two subwidgets, unfortunatelly
|
|
|
|
// the kill focus event is not propagated to the encompassing widget,
|
|
|
|
// so we need to bind it on the inner text widget instead. (Ugh.)
|
|
|
|
#ifdef __WXOSX__
|
|
|
|
temp->GetText()->Bind(wxEVT_KILL_FOCUS, ([this](wxEvent& e)
|
|
|
|
#else
|
2018-12-11 12:34:37 +00:00
|
|
|
temp->Bind(wxEVT_KILL_FOCUS, ([this](wxEvent& e)
|
2019-05-20 11:32:36 +00:00
|
|
|
#endif
|
2018-12-11 12:34:37 +00:00
|
|
|
{
|
2019-01-08 15:05:49 +00:00
|
|
|
e.Skip();
|
|
|
|
if (bEnterPressed) {
|
|
|
|
bEnterPressed = false;
|
|
|
|
return;
|
2018-12-11 12:34:37 +00:00
|
|
|
}
|
2019-01-08 15:05:49 +00:00
|
|
|
|
|
|
|
propagate_value();
|
2019-05-20 11:32:36 +00:00
|
|
|
}));
|
2018-12-18 12:22:22 +00:00
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
temp->Bind(wxEVT_SPINCTRL, ([this](wxCommandEvent e) { propagate_value(); }), temp->GetId());
|
|
|
|
|
2019-01-08 15:05:49 +00:00
|
|
|
temp->Bind(wxEVT_TEXT_ENTER, ([this](wxCommandEvent e)
|
|
|
|
{
|
|
|
|
e.Skip();
|
|
|
|
propagate_value();
|
|
|
|
bEnterPressed = true;
|
|
|
|
}), temp->GetId());
|
2018-12-21 09:58:00 +00:00
|
|
|
|
2018-01-31 15:46:17 +00:00
|
|
|
temp->Bind(wxEVT_TEXT, ([this](wxCommandEvent e)
|
2017-12-18 12:58:51 +00:00
|
|
|
{
|
|
|
|
// # 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
|
2019-05-20 11:32:36 +00:00
|
|
|
// # here temporarily so that we can return it from get_value()
|
2019-04-26 11:05:43 +00:00
|
|
|
|
2019-05-20 11:32:36 +00:00
|
|
|
long value;
|
|
|
|
const bool parsed = e.GetString().ToLong(&value);
|
2019-05-20 13:32:43 +00:00
|
|
|
tmp_value = parsed && value >= INT_MIN && value <= INT_MAX ? (int)value : UNDEF_VALUE;
|
2019-05-20 11:32:36 +00:00
|
|
|
|
|
|
|
#ifdef __WXOSX__
|
2021-02-24 08:22:31 +00:00
|
|
|
// Forcibly set the input value for SpinControl, since the value
|
2019-05-20 11:32:36 +00:00
|
|
|
// inserted from the keyboard or clipboard is not updated under OSX
|
|
|
|
if (tmp_value != UNDEF_VALUE) {
|
|
|
|
wxSpinCtrl* spin = static_cast<wxSpinCtrl*>(window);
|
2019-05-15 10:30:06 +00:00
|
|
|
spin->SetValue(tmp_value);
|
|
|
|
|
|
|
|
// But in SetValue() is executed m_text_ctrl->SelectAll(), so
|
|
|
|
// discard this selection and set insertion point to the end of string
|
|
|
|
spin->GetText()->SetInsertionPointEnd();
|
|
|
|
}
|
2018-12-18 12:22:22 +00:00
|
|
|
#endif
|
2017-12-18 12:58:51 +00:00
|
|
|
}), temp->GetId());
|
2021-02-24 08:22:31 +00:00
|
|
|
|
2018-01-27 13:21:16 +00:00
|
|
|
temp->SetToolTip(get_tooltip_text(text_value));
|
2017-12-18 12:58:51 +00:00
|
|
|
|
|
|
|
// recast as a wxWindow to fit the calling convention
|
|
|
|
window = dynamic_cast<wxWindow*>(temp);
|
|
|
|
}
|
|
|
|
|
2019-01-08 15:05:49 +00:00
|
|
|
void SpinCtrl::propagate_value()
|
|
|
|
{
|
2019-08-22 08:26:31 +00:00
|
|
|
if (suppress_propagation)
|
2019-08-22 08:25:19 +00:00
|
|
|
return;
|
|
|
|
|
2019-08-22 08:26:31 +00:00
|
|
|
suppress_propagation = true;
|
2019-05-20 11:32:36 +00:00
|
|
|
if (tmp_value == UNDEF_VALUE) {
|
2019-01-08 15:05:49 +00:00
|
|
|
on_kill_focus();
|
2019-05-20 11:32:36 +00:00
|
|
|
} else {
|
2019-08-19 10:58:59 +00:00
|
|
|
#ifdef __WXOSX__
|
|
|
|
// check input value for minimum
|
|
|
|
if (m_opt.min > 0 && tmp_value < m_opt.min) {
|
|
|
|
wxSpinCtrl* spin = static_cast<wxSpinCtrl*>(window);
|
|
|
|
spin->SetValue(m_opt.min);
|
|
|
|
spin->GetText()->SetInsertionPointEnd();
|
|
|
|
}
|
|
|
|
#endif
|
2019-01-08 15:05:49 +00:00
|
|
|
on_change_field();
|
2019-05-20 11:32:36 +00:00
|
|
|
}
|
2019-08-22 08:26:31 +00:00
|
|
|
suppress_propagation = false;
|
2019-01-08 15:05:49 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 14:28:43 +00:00
|
|
|
void SpinCtrl::msw_rescale()
|
2019-04-13 21:46:52 +00:00
|
|
|
{
|
2020-11-02 14:28:43 +00:00
|
|
|
Field::msw_rescale();
|
2019-04-13 21:46:52 +00:00
|
|
|
|
|
|
|
wxSpinCtrl* field = dynamic_cast<wxSpinCtrl*>(window);
|
2020-10-23 17:17:10 +00:00
|
|
|
if (parent_is_custom_ctrl)
|
|
|
|
field->SetSize(wxSize(def_width() * m_em_unit, lround(opt_height * m_em_unit)));
|
|
|
|
else
|
|
|
|
field->SetMinSize(wxSize(def_width() * m_em_unit, int(1.9f*field->GetFont().GetPixelSize().y)));
|
2019-04-13 21:46:52 +00:00
|
|
|
}
|
|
|
|
|
2020-10-16 19:18:23 +00:00
|
|
|
#ifdef __WXOSX__
|
2021-01-20 16:34:25 +00:00
|
|
|
static_assert(wxMAJOR_VERSION >= 3, "Use of wxBitmapComboBox on Settings Tabs requires wxWidgets 3.0 and newer");
|
2020-10-16 19:18:23 +00:00
|
|
|
using choice_ctrl = wxBitmapComboBox;
|
|
|
|
#else
|
|
|
|
using choice_ctrl = wxComboBox;
|
|
|
|
#endif // __WXOSX__
|
|
|
|
|
2017-12-18 12:58:51 +00:00
|
|
|
void Choice::BUILD() {
|
2020-02-12 11:08:43 +00:00
|
|
|
wxSize size(def_width_wider() * m_em_unit, wxDefaultCoord);
|
2019-04-17 19:35:53 +00:00
|
|
|
if (m_opt.height >= 0) size.SetHeight(m_opt.height*m_em_unit);
|
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width*m_em_unit);
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
choice_ctrl* temp;
|
2019-03-07 15:36:39 +00:00
|
|
|
if (!m_opt.gui_type.empty() && m_opt.gui_type.compare("select_open") != 0) {
|
|
|
|
m_is_editable = true;
|
2020-10-16 19:18:23 +00:00
|
|
|
temp = new choice_ctrl(m_parent, wxID_ANY, wxString(""), wxDefaultPosition, size);
|
2019-03-07 15:36:39 +00:00
|
|
|
}
|
2019-03-20 10:54:48 +00:00
|
|
|
else {
|
|
|
|
#ifdef __WXOSX__
|
|
|
|
/* wxBitmapComboBox with wxCB_READONLY style return NULL for GetTextCtrl(),
|
|
|
|
* so ToolTip doesn't shown.
|
|
|
|
* Next workaround helps to solve this problem
|
|
|
|
*/
|
2020-10-16 19:18:23 +00:00
|
|
|
temp = new choice_ctrl();
|
2019-03-20 10:54:48 +00:00
|
|
|
temp->SetTextCtrlStyle(wxTE_READONLY);
|
|
|
|
temp->Create(m_parent, wxID_ANY, wxString(""), wxDefaultPosition, size, 0, nullptr);
|
|
|
|
#else
|
2020-10-16 19:18:23 +00:00
|
|
|
temp = new choice_ctrl(m_parent, wxID_ANY, wxString(""), wxDefaultPosition, size, 0, nullptr, wxCB_READONLY);
|
2019-03-20 10:54:48 +00:00
|
|
|
#endif //__WXOSX__
|
|
|
|
}
|
|
|
|
|
2020-12-17 15:51:25 +00:00
|
|
|
#ifdef __WXGTK3__
|
|
|
|
wxSize best_sz = temp->GetBestSize();
|
|
|
|
if (best_sz.x > size.x)
|
|
|
|
temp->SetSize(best_sz);
|
|
|
|
#endif //__WXGTK3__
|
|
|
|
|
2019-03-18 19:54:01 +00:00
|
|
|
temp->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-11-12 12:54:36 +00:00
|
|
|
if (!wxOSX) temp->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2017-12-22 10:50:28 +00:00
|
|
|
|
|
|
|
// recast as a wxWindow to fit the calling convention
|
2017-12-18 12:58:51 +00:00
|
|
|
window = dynamic_cast<wxWindow*>(temp);
|
|
|
|
|
2019-05-20 10:21:05 +00:00
|
|
|
if (! m_opt.enum_labels.empty() || ! m_opt.enum_values.empty()) {
|
|
|
|
if (m_opt.enum_labels.empty()) {
|
|
|
|
// Append non-localized enum_values
|
|
|
|
for (auto el : m_opt.enum_values)
|
|
|
|
temp->Append(el);
|
|
|
|
} else {
|
|
|
|
// Append localized enum_labels
|
|
|
|
for (auto el : m_opt.enum_labels)
|
|
|
|
temp->Append(_(el));
|
2018-03-13 15:14:36 +00:00
|
|
|
}
|
2017-12-18 12:58:51 +00:00
|
|
|
set_selection();
|
|
|
|
}
|
2019-03-27 12:18:27 +00:00
|
|
|
|
2020-11-27 21:41:23 +00:00
|
|
|
temp->Bind(wxEVT_MOUSEWHEEL, [this](wxMouseEvent& e) {
|
|
|
|
if (m_suppress_scroll && !m_is_dropped)
|
|
|
|
e.StopPropagation();
|
|
|
|
else
|
|
|
|
e.Skip();
|
|
|
|
});
|
|
|
|
temp->Bind(wxEVT_COMBOBOX_DROPDOWN, [this](wxCommandEvent&) { m_is_dropped = true; });
|
|
|
|
temp->Bind(wxEVT_COMBOBOX_CLOSEUP, [this](wxCommandEvent&) { m_is_dropped = false; });
|
|
|
|
|
2020-11-28 00:17:01 +00:00
|
|
|
temp->Bind(wxEVT_COMBOBOX, [this](wxCommandEvent&) { on_change_field(); }, temp->GetId());
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2019-03-07 15:36:39 +00:00
|
|
|
if (m_is_editable) {
|
2018-12-18 12:22:22 +00:00
|
|
|
temp->Bind(wxEVT_KILL_FOCUS, ([this](wxEvent& e) {
|
|
|
|
e.Skip();
|
2020-11-28 00:17:01 +00:00
|
|
|
if (m_opt.type == coStrings) {
|
2019-05-15 08:41:14 +00:00
|
|
|
on_change_field();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-10-16 19:18:23 +00:00
|
|
|
if (is_defined_input_value<choice_ctrl>(window, m_opt.type)) {
|
2020-11-28 00:17:01 +00:00
|
|
|
if (m_opt.type == coFloatOrPercent) {
|
|
|
|
std::string old_val = !m_value.empty() ? boost::any_cast<std::string>(m_value) : "";
|
|
|
|
if (old_val == boost::any_cast<std::string>(get_value()))
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
double old_val = !m_value.empty() ? boost::any_cast<double>(m_value) : -99999;
|
|
|
|
if (fabs(old_val - boost::any_cast<double>(get_value())) <= 0.0001)
|
|
|
|
return;
|
|
|
|
}
|
2020-11-24 17:44:13 +00:00
|
|
|
on_change_field();
|
2018-12-18 12:22:22 +00:00
|
|
|
}
|
|
|
|
else
|
2019-01-08 15:05:49 +00:00
|
|
|
on_kill_focus();
|
2018-12-18 12:22:22 +00:00
|
|
|
}), temp->GetId());
|
|
|
|
}
|
|
|
|
|
2018-01-27 13:21:16 +00:00
|
|
|
temp->SetToolTip(get_tooltip_text(temp->GetValue()));
|
2017-12-18 12:58:51 +00:00
|
|
|
}
|
|
|
|
|
2020-11-24 17:44:13 +00:00
|
|
|
void Choice::suppress_scroll()
|
|
|
|
{
|
|
|
|
m_suppress_scroll = true;
|
|
|
|
}
|
|
|
|
|
2017-12-18 12:58:51 +00:00
|
|
|
void Choice::set_selection()
|
|
|
|
{
|
2019-03-27 15:59:34 +00:00
|
|
|
/* To prevent earlier control updating under OSX set m_disable_change_event to true
|
|
|
|
* (under OSX wxBitmapComboBox send wxEVT_COMBOBOX even after SetSelection())
|
|
|
|
*/
|
|
|
|
m_disable_change_event = true;
|
|
|
|
|
2017-12-18 12:58:51 +00:00
|
|
|
wxString text_value = wxString("");
|
2019-03-07 15:36:39 +00:00
|
|
|
|
2020-10-16 19:18:23 +00:00
|
|
|
choice_ctrl* field = dynamic_cast<choice_ctrl*>(window);
|
2018-10-31 11:56:08 +00:00
|
|
|
switch (m_opt.type) {
|
2020-11-18 16:00:08 +00:00
|
|
|
case coEnum:{
|
|
|
|
int id_value = m_opt.get_default_value<ConfigOptionEnum<SeamPosition>>()->value; //!!
|
|
|
|
field->SetSelection(id_value);
|
|
|
|
break;
|
|
|
|
}
|
2017-12-18 12:58:51 +00:00
|
|
|
case coFloat:
|
|
|
|
case coPercent: {
|
2018-01-05 14:11:33 +00:00
|
|
|
double val = m_opt.default_value->getFloat();
|
2017-12-18 12:58:51 +00:00
|
|
|
text_value = val - int(val) == 0 ? wxString::Format(_T("%i"), int(val)) : wxNumberFormatter::ToString(val, 1);
|
|
|
|
break;
|
|
|
|
}
|
2017-12-22 10:50:28 +00:00
|
|
|
case coInt:{
|
2020-11-18 16:00:08 +00:00
|
|
|
text_value = wxString::Format(_T("%i"), int(m_opt.default_value->getInt()));
|
2017-12-22 10:50:28 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case coStrings:{
|
2019-05-03 16:30:58 +00:00
|
|
|
text_value = m_opt.get_default_value<ConfigOptionStrings>()->get_at(m_opt_idx);
|
2020-11-18 16:00:08 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case coFloatOrPercent: {
|
|
|
|
text_value = double_to_string(m_opt.default_value->getFloat());
|
|
|
|
if (m_opt.get_default_value<ConfigOptionFloatOrPercent>()->percent)
|
|
|
|
text_value += "%";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default: break;
|
|
|
|
}
|
2017-12-22 10:50:28 +00:00
|
|
|
|
2020-11-18 16:00:08 +00:00
|
|
|
if (!text_value.IsEmpty()) {
|
2021-01-14 12:00:03 +00:00
|
|
|
size_t idx = 0;
|
2020-11-18 16:00:08 +00:00
|
|
|
for (auto el : m_opt.enum_values) {
|
2019-09-03 08:59:04 +00:00
|
|
|
if (el == text_value)
|
2017-12-22 10:50:28 +00:00
|
|
|
break;
|
|
|
|
++idx;
|
|
|
|
}
|
2020-11-18 16:00:08 +00:00
|
|
|
idx == m_opt.enum_values.size() ? field->SetValue(text_value) : field->SetSelection(idx);
|
2017-12-18 12:58:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-13 10:35:04 +00:00
|
|
|
void Choice::set_value(const std::string& value, bool change_event) //! Redundant?
|
2017-12-18 12:58:51 +00:00
|
|
|
{
|
2018-03-15 17:06:26 +00:00
|
|
|
m_disable_change_event = !change_event;
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2018-02-01 08:29:07 +00:00
|
|
|
size_t idx=0;
|
2018-01-05 14:11:33 +00:00
|
|
|
for (auto el : m_opt.enum_values)
|
2017-12-18 12:58:51 +00:00
|
|
|
{
|
2019-09-03 08:59:04 +00:00
|
|
|
if (el == value)
|
2017-12-18 12:58:51 +00:00
|
|
|
break;
|
|
|
|
++idx;
|
|
|
|
}
|
|
|
|
|
2020-10-16 19:18:23 +00:00
|
|
|
choice_ctrl* field = dynamic_cast<choice_ctrl*>(window);
|
2021-02-24 08:22:31 +00:00
|
|
|
idx == m_opt.enum_values.size() ?
|
2019-03-07 15:36:39 +00:00
|
|
|
field->SetValue(value) :
|
|
|
|
field->SetSelection(idx);
|
2021-02-24 08:22:31 +00:00
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
m_disable_change_event = false;
|
2017-12-18 12:58:51 +00:00
|
|
|
}
|
|
|
|
|
2018-04-13 10:35:04 +00:00
|
|
|
void Choice::set_value(const boost::any& value, bool change_event)
|
2018-01-09 08:41:07 +00:00
|
|
|
{
|
2018-03-15 17:06:26 +00:00
|
|
|
m_disable_change_event = !change_event;
|
2018-01-27 16:39:00 +00:00
|
|
|
|
2020-10-16 19:18:23 +00:00
|
|
|
choice_ctrl* field = dynamic_cast<choice_ctrl*>(window);
|
2019-03-07 15:36:39 +00:00
|
|
|
|
2018-10-31 11:56:08 +00:00
|
|
|
switch (m_opt.type) {
|
2018-01-09 08:41:07 +00:00
|
|
|
case coInt:
|
|
|
|
case coFloat:
|
|
|
|
case coPercent:
|
2020-11-18 16:00:08 +00:00
|
|
|
case coFloatOrPercent:
|
2018-03-23 11:52:37 +00:00
|
|
|
case coString:
|
2018-10-31 11:56:08 +00:00
|
|
|
case coStrings: {
|
2018-01-16 15:28:01 +00:00
|
|
|
wxString text_value;
|
2021-02-24 08:22:31 +00:00
|
|
|
if (m_opt.type == coInt)
|
2018-01-16 15:28:01 +00:00
|
|
|
text_value = wxString::Format(_T("%i"), int(boost::any_cast<int>(value)));
|
|
|
|
else
|
|
|
|
text_value = boost::any_cast<wxString>(value);
|
2019-09-06 15:46:55 +00:00
|
|
|
size_t idx = 0;
|
2020-11-25 18:18:44 +00:00
|
|
|
const std::vector<std::string>& enums = m_opt.enum_values.empty() ? m_opt.enum_labels : m_opt.enum_values;
|
|
|
|
for (auto el : enums)
|
2018-01-09 08:41:07 +00:00
|
|
|
{
|
2019-09-03 08:59:04 +00:00
|
|
|
if (el == text_value)
|
2018-01-09 08:41:07 +00:00
|
|
|
break;
|
|
|
|
++idx;
|
|
|
|
}
|
2020-11-25 18:18:44 +00:00
|
|
|
if (idx == enums.size()) {
|
2019-02-25 09:14:49 +00:00
|
|
|
// For editable Combobox under OSX is needed to set selection to -1 explicitly,
|
|
|
|
// otherwise selection doesn't be changed
|
2019-03-07 15:36:39 +00:00
|
|
|
field->SetSelection(-1);
|
|
|
|
field->SetValue(text_value);
|
2019-02-25 09:14:49 +00:00
|
|
|
}
|
|
|
|
else
|
2019-03-07 15:36:39 +00:00
|
|
|
field->SetSelection(idx);
|
2018-01-09 08:41:07 +00:00
|
|
|
break;
|
|
|
|
}
|
2018-10-31 11:56:08 +00:00
|
|
|
case coEnum: {
|
2018-04-24 08:33:11 +00:00
|
|
|
int val = boost::any_cast<int>(value);
|
2020-04-28 16:00:42 +00:00
|
|
|
if (m_opt_id == "top_fill_pattern" || m_opt_id == "bottom_fill_pattern" || m_opt_id == "fill_pattern")
|
2018-04-24 08:33:11 +00:00
|
|
|
{
|
2018-10-31 11:56:08 +00:00
|
|
|
if (!m_opt.enum_values.empty()) {
|
2018-04-24 08:33:11 +00:00
|
|
|
std::string key;
|
2021-02-24 08:22:31 +00:00
|
|
|
t_config_enum_values map_names = ConfigOptionEnum<InfillPattern>::get_enum_values();
|
2018-04-24 08:33:11 +00:00
|
|
|
for (auto it : map_names) {
|
|
|
|
if (val == it.second) {
|
|
|
|
key = it.first;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t idx = 0;
|
|
|
|
for (auto el : m_opt.enum_values)
|
|
|
|
{
|
2019-09-03 08:59:04 +00:00
|
|
|
if (el == key)
|
2018-04-24 08:33:11 +00:00
|
|
|
break;
|
|
|
|
++idx;
|
|
|
|
}
|
|
|
|
|
|
|
|
val = idx == m_opt.enum_values.size() ? 0 : idx;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
val = 0;
|
|
|
|
}
|
2019-03-07 15:36:39 +00:00
|
|
|
field->SetSelection(val);
|
2018-01-09 08:41:07 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2018-01-27 16:39:00 +00:00
|
|
|
|
|
|
|
m_disable_change_event = false;
|
2018-01-09 08:41:07 +00:00
|
|
|
}
|
|
|
|
|
2017-12-18 12:58:51 +00:00
|
|
|
//! it's needed for _update_serial_ports()
|
2018-04-13 10:35:04 +00:00
|
|
|
void Choice::set_values(const std::vector<std::string>& values)
|
2017-12-18 12:58:51 +00:00
|
|
|
{
|
2018-01-05 14:11:33 +00:00
|
|
|
if (values.empty())
|
|
|
|
return;
|
|
|
|
m_disable_change_event = true;
|
2017-12-18 12:58:51 +00:00
|
|
|
|
2018-03-15 17:06:26 +00:00
|
|
|
// # it looks that Clear() also clears the text field in recent wxWidgets versions,
|
2017-12-18 12:58:51 +00:00
|
|
|
// # but we want to preserve it
|
2020-10-16 19:18:23 +00:00
|
|
|
auto ww = dynamic_cast<choice_ctrl*>(window);
|
2017-12-18 12:58:51 +00:00
|
|
|
auto value = ww->GetValue();
|
|
|
|
ww->Clear();
|
2018-03-23 11:52:37 +00:00
|
|
|
ww->Append("");
|
2020-10-28 08:51:05 +00:00
|
|
|
for (const auto &el : values)
|
2017-12-18 12:58:51 +00:00
|
|
|
ww->Append(wxString(el));
|
|
|
|
ww->SetValue(value);
|
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
m_disable_change_event = false;
|
2017-12-18 12:58:51 +00:00
|
|
|
}
|
|
|
|
|
2020-10-28 08:51:05 +00:00
|
|
|
void Choice::set_values(const wxArrayString &values)
|
|
|
|
{
|
|
|
|
if (values.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_disable_change_event = true;
|
|
|
|
|
|
|
|
// # it looks that Clear() also clears the text field in recent wxWidgets versions,
|
|
|
|
// # but we want to preserve it
|
2020-11-14 20:46:54 +00:00
|
|
|
auto ww = dynamic_cast<choice_ctrl*>(window);
|
2020-10-28 08:51:05 +00:00
|
|
|
auto value = ww->GetValue();
|
|
|
|
ww->Clear();
|
|
|
|
ww->Append("");
|
|
|
|
for (const auto &el : values)
|
|
|
|
ww->Append(el);
|
|
|
|
ww->SetValue(value);
|
|
|
|
|
|
|
|
m_disable_change_event = false;
|
|
|
|
}
|
|
|
|
|
2018-04-24 08:33:11 +00:00
|
|
|
boost::any& Choice::get_value()
|
2018-01-09 12:52:01 +00:00
|
|
|
{
|
2020-10-16 19:18:23 +00:00
|
|
|
choice_ctrl* field = dynamic_cast<choice_ctrl*>(window);
|
2019-03-07 15:36:39 +00:00
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
wxString ret_str = field->GetValue();
|
2018-01-09 12:52:01 +00:00
|
|
|
|
2018-07-04 12:52:36 +00:00
|
|
|
// options from right panel
|
2019-08-08 07:48:56 +00:00
|
|
|
std::vector <std::string> right_panel_options{ "support", "pad", "scale_unit" };
|
2018-07-04 12:52:36 +00:00
|
|
|
for (auto rp_option: right_panel_options)
|
|
|
|
if (m_opt_id == rp_option)
|
|
|
|
return m_value = boost::any(ret_str);
|
2018-03-13 15:14:36 +00:00
|
|
|
|
2018-12-17 17:07:31 +00:00
|
|
|
if (m_opt.type == coEnum)
|
2018-01-11 09:33:17 +00:00
|
|
|
{
|
2021-02-24 08:22:31 +00:00
|
|
|
int ret_enum = field->GetSelection();
|
2020-04-28 16:00:42 +00:00
|
|
|
if (m_opt_id == "top_fill_pattern" || m_opt_id == "bottom_fill_pattern" || m_opt_id == "fill_pattern")
|
2018-01-25 12:46:04 +00:00
|
|
|
{
|
2018-10-31 11:56:08 +00:00
|
|
|
if (!m_opt.enum_values.empty()) {
|
2018-01-25 12:46:04 +00:00
|
|
|
std::string key = m_opt.enum_values[ret_enum];
|
|
|
|
t_config_enum_values map_names = ConfigOptionEnum<InfillPattern>::get_enum_values();
|
|
|
|
int value = map_names.at(key);
|
|
|
|
|
2018-04-24 08:33:11 +00:00
|
|
|
m_value = static_cast<InfillPattern>(value);
|
2018-01-25 12:46:04 +00:00
|
|
|
}
|
|
|
|
else
|
2018-04-24 08:33:11 +00:00
|
|
|
m_value = static_cast<InfillPattern>(0);
|
2018-01-25 12:46:04 +00:00
|
|
|
}
|
2020-04-14 09:53:28 +00:00
|
|
|
else if (m_opt_id.compare("ironing_type") == 0)
|
|
|
|
m_value = static_cast<IroningType>(ret_enum);
|
2021-02-10 15:02:32 +00:00
|
|
|
else if (m_opt_id.compare("fuzzy_skin") == 0)
|
|
|
|
m_value = static_cast<FuzzySkinType>(ret_enum);
|
2018-01-11 09:33:17 +00:00
|
|
|
else if (m_opt_id.compare("gcode_flavor") == 0)
|
2018-04-24 08:33:11 +00:00
|
|
|
m_value = static_cast<GCodeFlavor>(ret_enum);
|
2020-10-02 15:31:55 +00:00
|
|
|
else if (m_opt_id.compare("machine_limits_usage") == 0)
|
|
|
|
m_value = static_cast<MachineLimitsUsage>(ret_enum);
|
2018-01-11 09:33:17 +00:00
|
|
|
else if (m_opt_id.compare("support_material_pattern") == 0)
|
2018-04-24 08:33:11 +00:00
|
|
|
m_value = static_cast<SupportMaterialPattern>(ret_enum);
|
2021-02-24 14:59:09 +00:00
|
|
|
else if (m_opt_id.compare("support_material_interface_pattern") == 0)
|
|
|
|
m_value = static_cast<SupportMaterialInterfacePattern>(ret_enum);
|
2018-01-11 09:33:17 +00:00
|
|
|
else if (m_opt_id.compare("seam_position") == 0)
|
2018-04-24 08:33:11 +00:00
|
|
|
m_value = static_cast<SeamPosition>(ret_enum);
|
2018-07-08 12:32:48 +00:00
|
|
|
else if (m_opt_id.compare("host_type") == 0)
|
|
|
|
m_value = static_cast<PrintHostType>(ret_enum);
|
2018-12-14 16:19:17 +00:00
|
|
|
else if (m_opt_id.compare("display_orientation") == 0)
|
|
|
|
m_value = static_cast<SLADisplayOrientation>(ret_enum);
|
2019-01-09 11:21:43 +00:00
|
|
|
else if (m_opt_id.compare("support_pillar_connection_mode") == 0)
|
|
|
|
m_value = static_cast<SLAPillarConnectionMode>(ret_enum);
|
2020-09-08 13:30:59 +00:00
|
|
|
else if (m_opt_id == "printhost_authorization_type")
|
2020-06-25 10:58:59 +00:00
|
|
|
m_value = static_cast<AuthorizationType>(ret_enum);
|
2021-02-03 14:12:53 +00:00
|
|
|
else if (m_opt_id == "brim_type")
|
|
|
|
m_value = static_cast<BrimType>(ret_enum);
|
2018-12-17 17:07:31 +00:00
|
|
|
}
|
|
|
|
else if (m_opt.gui_type == "f_enum_open") {
|
2019-03-07 15:36:39 +00:00
|
|
|
const int ret_enum = field->GetSelection();
|
|
|
|
if (ret_enum < 0 || m_opt.enum_values.empty() || m_opt.type == coStrings ||
|
2019-05-20 10:21:05 +00:00
|
|
|
(ret_str != m_opt.enum_values[ret_enum] && ret_str != _(m_opt.enum_labels[ret_enum])))
|
2019-01-29 09:21:21 +00:00
|
|
|
// modifies ret_string!
|
2018-12-17 17:07:31 +00:00
|
|
|
get_value_by_opt_type(ret_str);
|
2020-11-18 16:00:08 +00:00
|
|
|
else if (m_opt.type == coFloatOrPercent)
|
|
|
|
m_value = m_opt.enum_values[ret_enum];
|
|
|
|
else
|
2018-12-18 12:22:22 +00:00
|
|
|
m_value = atof(m_opt.enum_values[ret_enum].c_str());
|
2018-12-17 17:07:31 +00:00
|
|
|
}
|
2021-02-24 08:22:31 +00:00
|
|
|
else
|
2019-01-29 09:21:21 +00:00
|
|
|
// modifies ret_string!
|
2018-12-17 17:07:31 +00:00
|
|
|
get_value_by_opt_type(ret_str);
|
2018-01-09 12:52:01 +00:00
|
|
|
|
2018-04-24 08:33:11 +00:00
|
|
|
return m_value;
|
2018-01-09 12:52:01 +00:00
|
|
|
}
|
|
|
|
|
2020-10-16 19:18:23 +00:00
|
|
|
void Choice::enable() { dynamic_cast<choice_ctrl*>(window)->Enable(); };
|
|
|
|
void Choice::disable() { dynamic_cast<choice_ctrl*>(window)->Disable(); };
|
|
|
|
|
2020-11-02 14:28:43 +00:00
|
|
|
void Choice::msw_rescale()
|
2019-04-13 21:46:52 +00:00
|
|
|
{
|
2019-04-24 23:45:00 +00:00
|
|
|
Field::msw_rescale();
|
2019-04-13 21:46:52 +00:00
|
|
|
|
2020-10-16 19:18:23 +00:00
|
|
|
choice_ctrl* field = dynamic_cast<choice_ctrl*>(window);
|
2020-10-23 17:17:10 +00:00
|
|
|
#ifdef __WXOSX__
|
2020-09-23 07:35:30 +00:00
|
|
|
const wxString selection = field->GetValue();// field->GetString(index);
|
2019-04-13 21:46:52 +00:00
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
/* To correct scaling (set new controll size) of a wxBitmapCombobox
|
|
|
|
* we need to refill control with new bitmaps. So, in our case :
|
2020-10-23 17:17:10 +00:00
|
|
|
* 1. clear control
|
2019-04-13 21:46:52 +00:00
|
|
|
* 2. add content
|
|
|
|
* 3. add scaled "empty" bitmap to the at least one item
|
|
|
|
*/
|
|
|
|
field->Clear();
|
|
|
|
wxSize size(wxDefaultSize);
|
2020-02-12 11:08:43 +00:00
|
|
|
size.SetWidth((m_opt.width > 0 ? m_opt.width : def_width_wider()) * m_em_unit);
|
2021-02-24 08:22:31 +00:00
|
|
|
|
2019-04-25 13:06:44 +00:00
|
|
|
// Set rescaled min height to correct layout
|
|
|
|
field->SetMinSize(wxSize(-1, int(1.5f*field->GetFont().GetPixelSize().y + 0.5f)));
|
|
|
|
// Set rescaled size
|
2019-04-13 21:46:52 +00:00
|
|
|
field->SetSize(size);
|
|
|
|
|
2019-05-20 10:21:05 +00:00
|
|
|
size_t idx = 0;
|
|
|
|
if (! m_opt.enum_labels.empty() || ! m_opt.enum_values.empty()) {
|
|
|
|
size_t counter = 0;
|
|
|
|
bool labels = ! m_opt.enum_labels.empty();
|
|
|
|
for (const std::string &el : labels ? m_opt.enum_labels : m_opt.enum_values) {
|
2019-05-20 10:22:08 +00:00
|
|
|
wxString text = labels ? _(el) : wxString::FromUTF8(el.c_str());
|
2019-05-20 10:21:05 +00:00
|
|
|
field->Append(text);
|
|
|
|
if (text == selection)
|
2019-04-13 21:46:52 +00:00
|
|
|
idx = counter;
|
2019-05-20 10:21:05 +00:00
|
|
|
++ counter;
|
2019-04-13 21:46:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
idx == m_opt.enum_values.size() ?
|
|
|
|
field->SetValue(selection) :
|
|
|
|
field->SetSelection(idx);
|
2020-10-23 17:17:10 +00:00
|
|
|
#else
|
|
|
|
auto size = wxSize(def_width_wider() * m_em_unit, wxDefaultCoord);
|
|
|
|
if (m_opt.height >= 0) size.SetHeight(m_opt.height * m_em_unit);
|
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width * m_em_unit);
|
|
|
|
|
|
|
|
if (parent_is_custom_ctrl)
|
|
|
|
field->SetSize(size);
|
|
|
|
else
|
|
|
|
field->SetMinSize(size);
|
|
|
|
#endif
|
2019-04-13 21:46:52 +00:00
|
|
|
}
|
|
|
|
|
2017-12-22 10:50:28 +00:00
|
|
|
void ColourPicker::BUILD()
|
|
|
|
{
|
2020-02-12 11:08:43 +00:00
|
|
|
auto size = wxSize(def_width() * m_em_unit, wxDefaultCoord);
|
2019-04-17 19:35:53 +00:00
|
|
|
if (m_opt.height >= 0) size.SetHeight(m_opt.height*m_em_unit);
|
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width*m_em_unit);
|
2017-12-22 10:50:28 +00:00
|
|
|
|
2018-10-18 09:32:21 +00:00
|
|
|
// Validate the color
|
2019-05-03 16:30:58 +00:00
|
|
|
wxString clr_str(m_opt.get_default_value<ConfigOptionStrings>()->get_at(m_opt_idx));
|
2018-10-18 09:32:21 +00:00
|
|
|
wxColour clr(clr_str);
|
2019-08-13 07:37:44 +00:00
|
|
|
if (clr_str.IsEmpty() || !clr.IsOk()) {
|
2018-10-18 09:32:21 +00:00
|
|
|
clr = wxTransparentColour;
|
|
|
|
}
|
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
auto temp = new wxColourPickerCtrl(m_parent, wxID_ANY, clr, wxDefaultPosition, size);
|
2020-10-23 17:17:10 +00:00
|
|
|
if (parent_is_custom_ctrl && m_opt.height < 0)
|
|
|
|
opt_height = (double)temp->GetSize().GetHeight() / m_em_unit;
|
2019-08-13 07:37:44 +00:00
|
|
|
temp->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-11-12 12:54:36 +00:00
|
|
|
if (!wxOSX) temp->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2018-10-04 09:12:55 +00:00
|
|
|
|
2017-12-22 10:50:28 +00:00
|
|
|
// // recast as a wxWindow to fit the calling convention
|
|
|
|
window = dynamic_cast<wxWindow*>(temp);
|
|
|
|
|
2018-01-31 15:46:17 +00:00
|
|
|
temp->Bind(wxEVT_COLOURPICKER_CHANGED, ([this](wxCommandEvent e) { on_change_field(); }), temp->GetId());
|
2017-12-22 10:50:28 +00:00
|
|
|
|
2018-10-18 09:32:21 +00:00
|
|
|
temp->SetToolTip(get_tooltip_text(clr_str));
|
2017-12-22 10:50:28 +00:00
|
|
|
}
|
|
|
|
|
2019-08-13 07:37:44 +00:00
|
|
|
void ColourPicker::set_undef_value(wxColourPickerCtrl* field)
|
2018-10-31 11:56:08 +00:00
|
|
|
{
|
2019-08-13 07:37:44 +00:00
|
|
|
field->SetColour(wxTransparentColour);
|
2018-02-15 16:30:33 +00:00
|
|
|
|
2019-08-13 07:37:44 +00:00
|
|
|
wxButton* btn = dynamic_cast<wxButton*>(field->GetPickerCtrl());
|
|
|
|
wxBitmap bmp = btn->GetBitmap();
|
|
|
|
wxMemoryDC dc(bmp);
|
2019-08-23 13:32:47 +00:00
|
|
|
if (!dc.IsOk()) return;
|
2019-08-13 07:37:44 +00:00
|
|
|
dc.SetTextForeground(*wxWHITE);
|
|
|
|
dc.SetFont(wxGetApp().normal_font());
|
|
|
|
|
|
|
|
const wxRect rect = wxRect(0, 0, bmp.GetWidth(), bmp.GetHeight());
|
|
|
|
dc.DrawLabel("undef", rect, wxALIGN_CENTER_HORIZONTAL|wxALIGN_CENTER_VERTICAL);
|
|
|
|
|
|
|
|
dc.SelectObject(wxNullBitmap);
|
|
|
|
btn->SetBitmapLabel(bmp);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ColourPicker::set_value(const boost::any& value, bool change_event)
|
|
|
|
{
|
|
|
|
m_disable_change_event = !change_event;
|
|
|
|
const wxString clr_str(boost::any_cast<wxString>(value));
|
|
|
|
auto field = dynamic_cast<wxColourPickerCtrl*>(window);
|
|
|
|
|
|
|
|
wxColour clr(clr_str);
|
|
|
|
if (clr_str.IsEmpty() || !clr.IsOk())
|
|
|
|
set_undef_value(field);
|
|
|
|
else
|
|
|
|
field->SetColour(clr);
|
|
|
|
|
|
|
|
m_disable_change_event = false;
|
|
|
|
}
|
2018-02-15 16:30:33 +00:00
|
|
|
|
2019-08-13 07:37:44 +00:00
|
|
|
boost::any& ColourPicker::get_value()
|
|
|
|
{
|
|
|
|
auto colour = static_cast<wxColourPickerCtrl*>(window)->GetColour();
|
|
|
|
if (colour == wxTransparentColour)
|
|
|
|
m_value = std::string("");
|
|
|
|
else {
|
|
|
|
auto clr_str = wxString::Format(wxT("#%02X%02X%02X"), colour.Red(), colour.Green(), colour.Blue());
|
|
|
|
m_value = clr_str.ToStdString();
|
|
|
|
}
|
2018-04-24 08:33:11 +00:00
|
|
|
return m_value;
|
2018-02-15 16:30:33 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 14:28:43 +00:00
|
|
|
void ColourPicker::msw_rescale()
|
2019-08-13 07:37:44 +00:00
|
|
|
{
|
|
|
|
Field::msw_rescale();
|
|
|
|
|
2020-02-12 11:08:43 +00:00
|
|
|
wxColourPickerCtrl* field = dynamic_cast<wxColourPickerCtrl*>(window);
|
|
|
|
auto size = wxSize(def_width() * m_em_unit, wxDefaultCoord);
|
2021-02-24 08:22:31 +00:00
|
|
|
if (m_opt.height >= 0)
|
2020-10-23 17:17:10 +00:00
|
|
|
size.SetHeight(m_opt.height * m_em_unit);
|
|
|
|
else if (parent_is_custom_ctrl && opt_height > 0)
|
|
|
|
size.SetHeight(lround(opt_height * m_em_unit));
|
2021-01-14 12:00:03 +00:00
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width * m_em_unit);
|
2020-10-23 17:17:10 +00:00
|
|
|
if (parent_is_custom_ctrl)
|
|
|
|
field->SetSize(size);
|
|
|
|
else
|
|
|
|
field->SetMinSize(size);
|
2020-02-12 11:08:43 +00:00
|
|
|
|
2019-08-13 07:37:44 +00:00
|
|
|
if (field->GetColour() == wxTransparentColour)
|
|
|
|
set_undef_value(field);
|
|
|
|
}
|
|
|
|
|
2018-01-25 12:46:04 +00:00
|
|
|
void PointCtrl::BUILD()
|
2017-12-22 10:50:28 +00:00
|
|
|
{
|
|
|
|
auto temp = new wxBoxSizer(wxHORIZONTAL);
|
2019-02-06 08:49:32 +00:00
|
|
|
|
2019-04-17 19:35:53 +00:00
|
|
|
const wxSize field_size(4 * m_em_unit, -1);
|
2017-12-22 10:50:28 +00:00
|
|
|
|
2019-05-03 16:30:58 +00:00
|
|
|
auto default_pt = m_opt.get_default_value<ConfigOptionPoints>()->values.at(0);
|
2018-08-17 13:53:43 +00:00
|
|
|
double val = default_pt(0);
|
2018-02-12 14:29:21 +00:00
|
|
|
wxString X = val - int(val) == 0 ? wxString::Format(_T("%i"), int(val)) : wxNumberFormatter::ToString(val, 2, wxNumberFormatter::Style_None);
|
2018-08-17 13:53:43 +00:00
|
|
|
val = default_pt(1);
|
2018-02-12 14:29:21 +00:00
|
|
|
wxString Y = val - int(val) == 0 ? wxString::Format(_T("%i"), int(val)) : wxNumberFormatter::ToString(val, 2, wxNumberFormatter::Style_None);
|
2017-12-22 10:50:28 +00:00
|
|
|
|
2019-01-09 10:01:40 +00:00
|
|
|
x_textctrl = new wxTextCtrl(m_parent, wxID_ANY, X, wxDefaultPosition, field_size, wxTE_PROCESS_ENTER);
|
|
|
|
y_textctrl = new wxTextCtrl(m_parent, wxID_ANY, Y, wxDefaultPosition, field_size, wxTE_PROCESS_ENTER);
|
2020-10-23 17:17:10 +00:00
|
|
|
if (parent_is_custom_ctrl && m_opt.height < 0)
|
|
|
|
opt_height = (double)x_textctrl->GetSize().GetHeight() / m_em_unit;
|
|
|
|
|
2021-01-14 12:00:03 +00:00
|
|
|
x_textctrl->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-03-18 11:48:39 +00:00
|
|
|
x_textctrl->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2019-03-18 19:54:01 +00:00
|
|
|
y_textctrl->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-03-18 11:48:39 +00:00
|
|
|
y_textctrl->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2017-12-22 10:50:28 +00:00
|
|
|
|
2019-03-18 11:48:39 +00:00
|
|
|
auto static_text_x = new wxStaticText(m_parent, wxID_ANY, "x : ");
|
|
|
|
auto static_text_y = new wxStaticText(m_parent, wxID_ANY, " y : ");
|
2019-03-18 19:54:01 +00:00
|
|
|
static_text_x->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-03-18 11:48:39 +00:00
|
|
|
static_text_x->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2019-03-18 19:54:01 +00:00
|
|
|
static_text_y->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-03-18 11:48:39 +00:00
|
|
|
static_text_y->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
|
|
|
|
|
|
|
temp->Add(static_text_x, 0, wxALIGN_CENTER_VERTICAL, 0);
|
2017-12-22 10:50:28 +00:00
|
|
|
temp->Add(x_textctrl);
|
2019-03-18 11:48:39 +00:00
|
|
|
temp->Add(static_text_y, 0, wxALIGN_CENTER_VERTICAL, 0);
|
2017-12-22 10:50:28 +00:00
|
|
|
temp->Add(y_textctrl);
|
|
|
|
|
2019-01-08 15:05:49 +00:00
|
|
|
x_textctrl->Bind(wxEVT_TEXT_ENTER, ([this](wxCommandEvent e) { propagate_value(x_textctrl); }), x_textctrl->GetId());
|
|
|
|
y_textctrl->Bind(wxEVT_TEXT_ENTER, ([this](wxCommandEvent e) { propagate_value(y_textctrl); }), y_textctrl->GetId());
|
|
|
|
|
|
|
|
x_textctrl->Bind(wxEVT_KILL_FOCUS, ([this](wxEvent& e) { e.Skip(); propagate_value(x_textctrl); }), x_textctrl->GetId());
|
|
|
|
y_textctrl->Bind(wxEVT_KILL_FOCUS, ([this](wxEvent& e) { e.Skip(); propagate_value(y_textctrl); }), y_textctrl->GetId());
|
2017-12-22 10:50:28 +00:00
|
|
|
|
|
|
|
// // recast as a wxWindow to fit the calling convention
|
|
|
|
sizer = dynamic_cast<wxSizer*>(temp);
|
|
|
|
|
2018-01-27 13:21:16 +00:00
|
|
|
x_textctrl->SetToolTip(get_tooltip_text(X+", "+Y));
|
|
|
|
y_textctrl->SetToolTip(get_tooltip_text(X+", "+Y));
|
2017-12-22 10:50:28 +00:00
|
|
|
}
|
|
|
|
|
2020-11-02 14:28:43 +00:00
|
|
|
void PointCtrl::msw_rescale()
|
2019-04-17 19:35:53 +00:00
|
|
|
{
|
2019-04-24 23:45:00 +00:00
|
|
|
Field::msw_rescale();
|
2019-04-17 19:35:53 +00:00
|
|
|
|
2020-10-23 17:17:10 +00:00
|
|
|
wxSize field_size(4 * m_em_unit, -1);
|
2019-04-17 19:35:53 +00:00
|
|
|
|
2020-10-23 17:17:10 +00:00
|
|
|
if (parent_is_custom_ctrl) {
|
|
|
|
field_size.SetHeight(lround(opt_height * m_em_unit));
|
|
|
|
x_textctrl->SetSize(field_size);
|
|
|
|
y_textctrl->SetSize(field_size);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
x_textctrl->SetMinSize(field_size);
|
|
|
|
y_textctrl->SetMinSize(field_size);
|
|
|
|
}
|
2019-04-17 19:35:53 +00:00
|
|
|
}
|
|
|
|
|
2020-04-01 17:01:00 +00:00
|
|
|
bool PointCtrl::value_was_changed(wxTextCtrl* win)
|
|
|
|
{
|
|
|
|
if (m_value.empty())
|
|
|
|
return true;
|
|
|
|
|
|
|
|
boost::any val = m_value;
|
|
|
|
// update m_value!
|
|
|
|
get_value();
|
|
|
|
|
|
|
|
return boost::any_cast<Vec2d>(m_value) != boost::any_cast<Vec2d>(val);
|
|
|
|
}
|
|
|
|
|
2019-01-08 15:05:49 +00:00
|
|
|
void PointCtrl::propagate_value(wxTextCtrl* win)
|
2018-12-18 12:22:22 +00:00
|
|
|
{
|
2020-04-01 17:01:00 +00:00
|
|
|
if (win->GetValue().empty())
|
2019-01-08 15:05:49 +00:00
|
|
|
on_kill_focus();
|
2020-04-01 17:01:00 +00:00
|
|
|
else if (value_was_changed(win))
|
|
|
|
on_change_field();
|
2018-12-18 12:22:22 +00:00
|
|
|
}
|
|
|
|
|
2018-08-21 19:05:24 +00:00
|
|
|
void PointCtrl::set_value(const Vec2d& value, bool change_event)
|
2017-12-22 10:50:28 +00:00
|
|
|
{
|
2018-03-15 17:06:26 +00:00
|
|
|
m_disable_change_event = !change_event;
|
2017-12-22 10:50:28 +00:00
|
|
|
|
2018-08-17 13:53:43 +00:00
|
|
|
double val = value(0);
|
2018-02-12 14:29:21 +00:00
|
|
|
x_textctrl->SetValue(val - int(val) == 0 ? wxString::Format(_T("%i"), int(val)) : wxNumberFormatter::ToString(val, 2, wxNumberFormatter::Style_None));
|
2018-08-17 13:53:43 +00:00
|
|
|
val = value(1);
|
2018-02-12 14:29:21 +00:00
|
|
|
y_textctrl->SetValue(val - int(val) == 0 ? wxString::Format(_T("%i"), int(val)) : wxNumberFormatter::ToString(val, 2, wxNumberFormatter::Style_None));
|
2017-12-22 10:50:28 +00:00
|
|
|
|
2018-01-05 14:11:33 +00:00
|
|
|
m_disable_change_event = false;
|
2017-12-22 10:50:28 +00:00
|
|
|
}
|
|
|
|
|
2018-04-13 10:35:04 +00:00
|
|
|
void PointCtrl::set_value(const boost::any& value, bool change_event)
|
2018-01-25 12:46:04 +00:00
|
|
|
{
|
2018-08-21 19:05:24 +00:00
|
|
|
Vec2d pt(Vec2d::Zero());
|
|
|
|
const Vec2d *ptf = boost::any_cast<Vec2d>(&value);
|
2018-02-18 22:11:43 +00:00
|
|
|
if (!ptf)
|
2018-01-25 12:46:04 +00:00
|
|
|
{
|
2018-02-18 22:11:43 +00:00
|
|
|
ConfigOptionPoints* pts = boost::any_cast<ConfigOptionPoints*>(value);
|
|
|
|
pt = pts->values.at(0);
|
2018-01-25 12:46:04 +00:00
|
|
|
}
|
2018-02-18 22:11:43 +00:00
|
|
|
else
|
|
|
|
pt = *ptf;
|
2018-03-15 17:06:26 +00:00
|
|
|
set_value(pt, change_event);
|
2018-01-25 12:46:04 +00:00
|
|
|
}
|
|
|
|
|
2018-04-24 08:33:11 +00:00
|
|
|
boost::any& PointCtrl::get_value()
|
2017-12-22 10:50:28 +00:00
|
|
|
{
|
2018-08-21 18:34:45 +00:00
|
|
|
double x, y;
|
2020-04-05 21:18:22 +00:00
|
|
|
if (!x_textctrl->GetValue().ToDouble(&x) ||
|
|
|
|
!y_textctrl->GetValue().ToDouble(&y))
|
|
|
|
{
|
|
|
|
set_value(m_value.empty() ? Vec2d(0.0, 0.0) : m_value, true);
|
|
|
|
show_error(m_parent, _L("Invalid numeric input."));
|
|
|
|
}
|
|
|
|
else
|
2020-04-01 17:01:00 +00:00
|
|
|
if (m_opt.min > x || x > m_opt.max ||
|
|
|
|
m_opt.min > y || y > m_opt.max)
|
2021-02-24 08:22:31 +00:00
|
|
|
{
|
2020-04-01 17:01:00 +00:00
|
|
|
if (m_opt.min > x) x = m_opt.min;
|
|
|
|
if (x > m_opt.max) x = m_opt.max;
|
|
|
|
if (m_opt.min > y) y = m_opt.min;
|
|
|
|
if (y > m_opt.max) y = m_opt.max;
|
|
|
|
set_value(Vec2d(x, y), true);
|
|
|
|
|
2020-04-05 21:18:22 +00:00
|
|
|
show_error(m_parent, _L("Input value is out of range"));
|
2020-04-01 17:01:00 +00:00
|
|
|
}
|
|
|
|
|
2018-08-21 19:05:24 +00:00
|
|
|
return m_value = Vec2d(x, y);
|
2017-12-22 10:50:28 +00:00
|
|
|
}
|
|
|
|
|
2018-06-21 14:15:56 +00:00
|
|
|
void StaticText::BUILD()
|
|
|
|
{
|
|
|
|
auto size = wxSize(wxDefaultSize);
|
2019-04-17 19:35:53 +00:00
|
|
|
if (m_opt.height >= 0) size.SetHeight(m_opt.height*m_em_unit);
|
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width*m_em_unit);
|
2018-06-21 14:15:56 +00:00
|
|
|
|
2019-05-09 13:13:14 +00:00
|
|
|
const wxString legend = wxString::FromUTF8(m_opt.get_default_value<ConfigOptionString>()->value.c_str());
|
2019-02-07 15:33:14 +00:00
|
|
|
auto temp = new wxStaticText(m_parent, wxID_ANY, legend, wxDefaultPosition, size, wxST_ELLIPSIZE_MIDDLE);
|
2019-03-18 19:54:01 +00:00
|
|
|
temp->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-03-18 11:48:39 +00:00
|
|
|
temp->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2018-10-01 13:09:31 +00:00
|
|
|
temp->SetFont(wxGetApp().bold_font());
|
2018-06-21 14:15:56 +00:00
|
|
|
|
|
|
|
// // recast as a wxWindow to fit the calling convention
|
|
|
|
window = dynamic_cast<wxWindow*>(temp);
|
|
|
|
|
|
|
|
temp->SetToolTip(get_tooltip_text(legend));
|
|
|
|
}
|
|
|
|
|
2020-11-02 14:28:43 +00:00
|
|
|
void StaticText::msw_rescale()
|
2019-04-13 21:46:52 +00:00
|
|
|
{
|
2019-04-24 23:45:00 +00:00
|
|
|
Field::msw_rescale();
|
2019-04-13 21:46:52 +00:00
|
|
|
|
|
|
|
auto size = wxSize(wxDefaultSize);
|
2019-04-17 19:35:53 +00:00
|
|
|
if (m_opt.height >= 0) size.SetHeight(m_opt.height*m_em_unit);
|
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width*m_em_unit);
|
2019-04-13 21:46:52 +00:00
|
|
|
|
|
|
|
if (size != wxDefaultSize)
|
|
|
|
{
|
|
|
|
wxStaticText* field = dynamic_cast<wxStaticText*>(window);
|
|
|
|
field->SetSize(size);
|
2019-05-09 09:42:45 +00:00
|
|
|
field->SetMinSize(size);
|
2019-04-13 21:46:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-12 21:42:01 +00:00
|
|
|
void SliderCtrl::BUILD()
|
|
|
|
{
|
|
|
|
auto size = wxSize(wxDefaultSize);
|
|
|
|
if (m_opt.height >= 0) size.SetHeight(m_opt.height);
|
|
|
|
if (m_opt.width >= 0) size.SetWidth(m_opt.width);
|
|
|
|
|
|
|
|
auto temp = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
|
2019-05-03 16:30:58 +00:00
|
|
|
auto def_val = m_opt.get_default_value<ConfigOptionInt>()->value;
|
2018-06-12 21:42:01 +00:00
|
|
|
auto min = m_opt.min == INT_MIN ? 0 : m_opt.min;
|
|
|
|
auto max = m_opt.max == INT_MAX ? 100 : m_opt.max;
|
|
|
|
|
2018-06-15 20:42:51 +00:00
|
|
|
m_slider = new wxSlider(m_parent, wxID_ANY, def_val * m_scale,
|
2018-06-12 21:42:01 +00:00
|
|
|
min * m_scale, max * m_scale,
|
|
|
|
wxDefaultPosition, size);
|
2019-03-18 19:54:01 +00:00
|
|
|
m_slider->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-03-18 11:48:39 +00:00
|
|
|
m_slider->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2018-06-12 21:42:01 +00:00
|
|
|
wxSize field_size(40, -1);
|
|
|
|
|
2021-02-24 08:22:31 +00:00
|
|
|
m_textctrl = new wxTextCtrl(m_parent, wxID_ANY, wxString::Format("%d", m_slider->GetValue()/m_scale),
|
2018-06-12 21:42:01 +00:00
|
|
|
wxDefaultPosition, field_size);
|
2019-03-18 19:54:01 +00:00
|
|
|
m_textctrl->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
2019-03-18 11:48:39 +00:00
|
|
|
m_textctrl->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
2018-06-12 21:42:01 +00:00
|
|
|
|
|
|
|
temp->Add(m_slider, 1, wxEXPAND | wxALIGN_CENTER_VERTICAL, 0);
|
|
|
|
temp->Add(m_textctrl, 0, wxALIGN_CENTER_VERTICAL, 0);
|
|
|
|
|
|
|
|
m_slider->Bind(wxEVT_SLIDER, ([this](wxCommandEvent e) {
|
2018-10-31 11:56:08 +00:00
|
|
|
if (!m_disable_change_event) {
|
2018-06-12 21:42:01 +00:00
|
|
|
int val = boost::any_cast<int>(get_value());
|
|
|
|
m_textctrl->SetLabel(wxString::Format("%d", val));
|
|
|
|
on_change_field();
|
|
|
|
}
|
|
|
|
}), m_slider->GetId());
|
|
|
|
|
|
|
|
m_textctrl->Bind(wxEVT_TEXT, ([this](wxCommandEvent e) {
|
|
|
|
std::string value = e.GetString().utf8_str().data();
|
2018-10-31 11:56:08 +00:00
|
|
|
if (is_matched(value, "^-?\\d+(\\.\\d*)?$")) {
|
2018-06-12 21:42:01 +00:00
|
|
|
m_disable_change_event = true;
|
|
|
|
m_slider->SetValue(stoi(value)*m_scale);
|
|
|
|
m_disable_change_event = false;
|
|
|
|
on_change_field();
|
|
|
|
}
|
|
|
|
}), m_textctrl->GetId());
|
|
|
|
|
|
|
|
m_sizer = dynamic_cast<wxSizer*>(temp);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SliderCtrl::set_value(const boost::any& value, bool change_event)
|
|
|
|
{
|
|
|
|
m_disable_change_event = !change_event;
|
|
|
|
|
|
|
|
m_slider->SetValue(boost::any_cast<int>(value)*m_scale);
|
|
|
|
int val = boost::any_cast<int>(get_value());
|
|
|
|
m_textctrl->SetLabel(wxString::Format("%d", val));
|
|
|
|
|
|
|
|
m_disable_change_event = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
boost::any& SliderCtrl::get_value()
|
|
|
|
{
|
|
|
|
// int ret_val;
|
|
|
|
// x_textctrl->GetValue().ToDouble(&val);
|
|
|
|
return m_value = int(m_slider->GetValue()/m_scale);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-18 12:58:51 +00:00
|
|
|
} // GUI
|
|
|
|
} // Slic3r
|