2018-01-25 12:46:04 +00:00
|
|
|
#include "BedShapeDialog.hpp"
|
2020-05-26 15:42:57 +00:00
|
|
|
#include "GUI_App.hpp"
|
2020-05-27 11:19:26 +00:00
|
|
|
#include "OptionsGroup.hpp"
|
2018-01-25 12:46:04 +00:00
|
|
|
|
|
|
|
#include <wx/wx.h>
|
|
|
|
#include <wx/numformatter.h>
|
2018-11-26 13:41:58 +00:00
|
|
|
#include <wx/sizer.h>
|
|
|
|
#include <wx/statbox.h>
|
2019-07-25 11:42:42 +00:00
|
|
|
#include <wx/tooltip.h>
|
2018-11-26 13:41:58 +00:00
|
|
|
|
|
|
|
#include "libslic3r/BoundingBox.hpp"
|
|
|
|
#include "libslic3r/Model.hpp"
|
|
|
|
#include "libslic3r/Polygon.hpp"
|
|
|
|
|
2018-01-30 11:10:12 +00:00
|
|
|
#include "boost/nowide/iostream.hpp"
|
2019-07-18 09:12:11 +00:00
|
|
|
#include <boost/algorithm/string/predicate.hpp>
|
2019-11-12 11:29:31 +00:00
|
|
|
#include <boost/filesystem.hpp>
|
2018-01-25 12:46:04 +00:00
|
|
|
|
2018-07-30 08:03:17 +00:00
|
|
|
#include <algorithm>
|
|
|
|
|
2018-01-25 12:46:04 +00:00
|
|
|
namespace Slic3r {
|
|
|
|
namespace GUI {
|
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
BedShape::BedShape(const ConfigOptionPoints& points)
|
2018-01-25 12:46:04 +00:00
|
|
|
{
|
2020-08-19 13:35:58 +00:00
|
|
|
auto polygon = Polygon::new_scale(points.values);
|
2018-01-25 12:46:04 +00:00
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
// is this a rectangle ?
|
|
|
|
if (points.size() == 4) {
|
|
|
|
auto lines = polygon.lines();
|
|
|
|
if (lines[0].parallel_to(lines[2]) && lines[1].parallel_to(lines[3])) {
|
|
|
|
// okay, it's a rectangle
|
|
|
|
// find origin
|
|
|
|
coordf_t x_min, x_max, y_min, y_max;
|
|
|
|
x_max = x_min = points.values[0](0);
|
|
|
|
y_max = y_min = points.values[0](1);
|
|
|
|
for (auto pt : points.values)
|
|
|
|
{
|
|
|
|
x_min = std::min(x_min, pt(0));
|
|
|
|
x_max = std::max(x_max, pt(0));
|
|
|
|
y_min = std::min(y_min, pt(1));
|
|
|
|
y_max = std::max(y_max, pt(1));
|
|
|
|
}
|
2018-01-25 12:46:04 +00:00
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
m_type = Type::Rectangular;
|
|
|
|
m_rectSize = Vec2d(x_max - x_min, y_max - y_min);
|
|
|
|
m_rectOrigin = Vec2d(-x_min, -y_min);
|
2019-04-18 00:03:40 +00:00
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2019-04-18 00:03:40 +00:00
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
// is this a circle ?
|
|
|
|
{
|
|
|
|
// Analyze the array of points.Do they reside on a circle ?
|
|
|
|
auto center = polygon.bounding_box().center();
|
|
|
|
std::vector<double> vertex_distances;
|
|
|
|
double avg_dist = 0;
|
|
|
|
for (auto pt : polygon.points)
|
|
|
|
{
|
|
|
|
double distance = (pt - center).cast<double>().norm();
|
|
|
|
vertex_distances.push_back(distance);
|
|
|
|
avg_dist += distance;
|
|
|
|
}
|
|
|
|
|
|
|
|
avg_dist /= vertex_distances.size();
|
|
|
|
bool defined_value = true;
|
|
|
|
for (auto el : vertex_distances)
|
|
|
|
{
|
|
|
|
if (abs(el - avg_dist) > 10 * SCALED_EPSILON)
|
|
|
|
defined_value = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (defined_value) {
|
|
|
|
// all vertices are equidistant to center
|
|
|
|
m_type = Type::Circular;
|
|
|
|
m_diameter = unscale<double>(avg_dist * 2);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2019-04-18 00:03:40 +00:00
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
if (points.size() < 3)
|
|
|
|
return;
|
2019-04-18 00:03:40 +00:00
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
// This is a custom bed shape, use the polygon provided.
|
|
|
|
m_type = Type::Custom;
|
2019-04-18 00:03:40 +00:00
|
|
|
}
|
|
|
|
|
2020-08-19 11:04:51 +00:00
|
|
|
static std::string get_option_label(BedShape::Parameter param)
|
|
|
|
{
|
|
|
|
switch (param) {
|
|
|
|
case BedShape::Parameter::RectSize : return L("Size");
|
|
|
|
case BedShape::Parameter::RectOrigin: return L("Origin");
|
|
|
|
case BedShape::Parameter::Diameter : return L("Diameter");
|
|
|
|
default: return "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void BedShape::append_option_line(ConfigOptionsGroupShp optgroup, Parameter param)
|
|
|
|
{
|
|
|
|
ConfigOptionDef def;
|
|
|
|
|
|
|
|
if (param == Parameter::RectSize) {
|
|
|
|
def.type = coPoints;
|
|
|
|
def.set_default_value(new ConfigOptionPoints{ Vec2d(200, 200) });
|
|
|
|
def.min = 0;
|
|
|
|
def.max = 1200;
|
|
|
|
def.label = get_option_label(param);
|
|
|
|
def.tooltip = L("Size in X and Y of the rectangular plate.");
|
|
|
|
|
|
|
|
Option option(def, "rect_size");
|
|
|
|
optgroup->append_single_option_line(option);
|
|
|
|
}
|
|
|
|
else if (param == Parameter::RectOrigin) {
|
|
|
|
def.type = coPoints;
|
|
|
|
def.set_default_value(new ConfigOptionPoints{ Vec2d(0, 0) });
|
|
|
|
def.min = -600;
|
|
|
|
def.max = 600;
|
|
|
|
def.label = get_option_label(param);
|
|
|
|
def.tooltip = L("Distance of the 0,0 G-code coordinate from the front left corner of the rectangle.");
|
|
|
|
|
|
|
|
Option option(def, "rect_origin");
|
|
|
|
optgroup->append_single_option_line(option);
|
|
|
|
}
|
|
|
|
else if (param == Parameter::Diameter) {
|
|
|
|
def.type = coFloat;
|
|
|
|
def.set_default_value(new ConfigOptionFloat(200));
|
|
|
|
def.sidetext = L("mm");
|
|
|
|
def.label = get_option_label(param);
|
|
|
|
def.tooltip = L("Diameter of the print bed. It is assumed that origin (0,0) is located in the center.");
|
|
|
|
|
|
|
|
Option option(def, "diameter");
|
|
|
|
optgroup->append_single_option_line(option);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
wxString BedShape::get_name(Type type)
|
|
|
|
{
|
|
|
|
switch (type) {
|
|
|
|
case Type::Rectangular : return _L("Rectangular");
|
|
|
|
case Type::Circular : return _L("Circular");
|
|
|
|
case Type::Custom : return _L("Custom");
|
|
|
|
case Type::Invalid :
|
|
|
|
default : return _L("Invalid");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
size_t BedShape::get_type()
|
|
|
|
{
|
|
|
|
return static_cast<size_t>(m_type == Type::Invalid ? Type::Rectangular : m_type);
|
|
|
|
}
|
|
|
|
|
2020-08-19 11:04:51 +00:00
|
|
|
wxString BedShape::get_full_name_with_params()
|
|
|
|
{
|
2020-08-19 13:35:58 +00:00
|
|
|
wxString out = _L("Shape") + ": " + get_name(m_type);
|
2020-08-19 11:04:51 +00:00
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
if (m_type == Type::Rectangular) {
|
|
|
|
out += "\n" + _(get_option_label(Parameter::RectSize)) + ": [" + ConfigOptionPoint(m_rectSize).serialize() + "]";
|
|
|
|
out += "\n" + _(get_option_label(Parameter::RectOrigin))+ ": [" + ConfigOptionPoint(m_rectOrigin).serialize() + "]";
|
2020-08-19 11:04:51 +00:00
|
|
|
}
|
2020-08-19 13:35:58 +00:00
|
|
|
else if (m_type == Type::Circular)
|
|
|
|
out += "\n" + _L(get_option_label(Parameter::Diameter)) + ": [" + double_to_string(m_diameter) + "]";
|
2020-08-19 11:04:51 +00:00
|
|
|
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
void BedShape::apply_optgroup_values(ConfigOptionsGroupShp optgroup)
|
|
|
|
{
|
|
|
|
if (m_type == Type::Rectangular || m_type == Type::Invalid) {
|
|
|
|
optgroup->set_value("rect_size" , new ConfigOptionPoints{ m_rectSize });
|
|
|
|
optgroup->set_value("rect_origin" , new ConfigOptionPoints{ m_rectOrigin });
|
|
|
|
}
|
|
|
|
else if (m_type == Type::Circular)
|
|
|
|
optgroup->set_value("diameter", double_to_string(m_diameter));
|
|
|
|
}
|
|
|
|
|
|
|
|
void BedShapeDialog::build_dialog(const ConfigOptionPoints& default_pt, const ConfigOptionString& custom_texture, const ConfigOptionString& custom_model)
|
|
|
|
{
|
|
|
|
SetFont(wxGetApp().normal_font());
|
|
|
|
m_panel = new BedShapePanel(this);
|
|
|
|
m_panel->build_panel(default_pt, custom_texture, custom_model);
|
|
|
|
|
|
|
|
auto main_sizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
main_sizer->Add(m_panel, 1, wxEXPAND);
|
|
|
|
main_sizer->Add(CreateButtonSizer(wxOK | wxCANCEL), 0, wxALIGN_CENTER_HORIZONTAL | wxBOTTOM, 10);
|
|
|
|
|
|
|
|
SetSizer(main_sizer);
|
|
|
|
SetMinSize(GetSize());
|
|
|
|
main_sizer->SetSizeHints(this);
|
|
|
|
|
|
|
|
this->Bind(wxEVT_CLOSE_WINDOW, ([this](wxCloseEvent& evt) {
|
|
|
|
EndModal(wxID_CANCEL);
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
void BedShapeDialog::on_dpi_changed(const wxRect &suggested_rect)
|
|
|
|
{
|
|
|
|
const int& em = em_unit();
|
|
|
|
m_panel->m_shape_options_book->SetMinSize(wxSize(25 * em, -1));
|
|
|
|
|
|
|
|
for (auto og : m_panel->m_optgroups)
|
|
|
|
og->msw_rescale();
|
|
|
|
|
|
|
|
const wxSize& size = wxSize(50 * em, -1);
|
|
|
|
|
|
|
|
SetMinSize(size);
|
|
|
|
SetSize(size);
|
|
|
|
|
|
|
|
Refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string BedShapePanel::NONE = "None";
|
|
|
|
const std::string BedShapePanel::EMPTY_STRING = "";
|
|
|
|
|
2019-07-18 10:56:52 +00:00
|
|
|
void BedShapePanel::build_panel(const ConfigOptionPoints& default_pt, const ConfigOptionString& custom_texture, const ConfigOptionString& custom_model)
|
2018-01-25 12:46:04 +00:00
|
|
|
{
|
2019-07-18 09:12:11 +00:00
|
|
|
m_shape = default_pt.values;
|
|
|
|
m_custom_texture = custom_texture.value.empty() ? NONE : custom_texture.value;
|
2019-07-18 10:56:52 +00:00
|
|
|
m_custom_model = custom_model.value.empty() ? NONE : custom_model.value;
|
2019-07-18 09:12:11 +00:00
|
|
|
|
2019-07-03 08:06:22 +00:00
|
|
|
auto sbsizer = new wxStaticBoxSizer(wxVERTICAL, this, _(L("Shape")));
|
2019-07-09 11:33:15 +00:00
|
|
|
sbsizer->GetStaticBox()->SetFont(wxGetApp().bold_font());
|
2018-01-25 12:46:04 +00:00
|
|
|
|
|
|
|
// shape options
|
2019-07-18 09:12:11 +00:00
|
|
|
m_shape_options_book = new wxChoicebook(this, wxID_ANY, wxDefaultPosition, wxSize(25*wxGetApp().em_unit(), -1), wxCHB_TOP);
|
2019-07-03 08:06:22 +00:00
|
|
|
sbsizer->Add(m_shape_options_book);
|
2018-01-25 12:46:04 +00:00
|
|
|
|
2020-08-19 11:04:51 +00:00
|
|
|
auto optgroup = init_shape_options_page(BedShape::get_name(BedShape::Type::Rectangular));
|
|
|
|
BedShape::append_option_line(optgroup, BedShape::Parameter::RectSize);
|
|
|
|
BedShape::append_option_line(optgroup, BedShape::Parameter::RectOrigin);
|
2020-09-25 08:44:22 +00:00
|
|
|
activate_options_page(optgroup);
|
2020-08-19 11:04:51 +00:00
|
|
|
|
|
|
|
optgroup = init_shape_options_page(BedShape::get_name(BedShape::Type::Circular));
|
|
|
|
BedShape::append_option_line(optgroup, BedShape::Parameter::Diameter);
|
2020-09-25 08:44:22 +00:00
|
|
|
activate_options_page(optgroup);
|
2020-08-19 11:04:51 +00:00
|
|
|
|
|
|
|
optgroup = init_shape_options_page(BedShape::get_name(BedShape::Type::Custom));
|
2019-07-09 11:33:15 +00:00
|
|
|
|
|
|
|
Line line{ "", "" };
|
|
|
|
line.full_width = 1;
|
|
|
|
line.widget = [this](wxWindow* parent) {
|
2020-08-19 11:04:51 +00:00
|
|
|
wxButton* shape_btn = new wxButton(parent, wxID_ANY, _L("Load shape from STL..."));
|
2019-07-09 11:33:15 +00:00
|
|
|
wxSizer* shape_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
shape_sizer->Add(shape_btn, 1, wxEXPAND);
|
|
|
|
|
|
|
|
wxSizer* sizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
sizer->Add(shape_sizer, 1, wxEXPAND);
|
|
|
|
|
2020-08-19 11:04:51 +00:00
|
|
|
shape_btn->Bind(wxEVT_BUTTON, [this](wxCommandEvent& e) {
|
2019-07-09 11:33:15 +00:00
|
|
|
load_stl();
|
2020-08-19 11:04:51 +00:00
|
|
|
});
|
2019-07-09 11:33:15 +00:00
|
|
|
|
|
|
|
return sizer;
|
|
|
|
};
|
|
|
|
optgroup->append_line(line);
|
2020-09-25 08:44:22 +00:00
|
|
|
activate_options_page(optgroup);
|
2018-01-25 12:46:04 +00:00
|
|
|
|
2019-07-18 09:12:11 +00:00
|
|
|
wxPanel* texture_panel = init_texture_panel();
|
2019-07-18 10:56:52 +00:00
|
|
|
wxPanel* model_panel = init_model_panel();
|
2019-07-18 09:12:11 +00:00
|
|
|
|
2019-07-03 08:06:22 +00:00
|
|
|
Bind(wxEVT_CHOICEBOOK_PAGE_CHANGED, ([this](wxCommandEvent& e)
|
|
|
|
{
|
2018-01-25 12:46:04 +00:00
|
|
|
update_shape();
|
|
|
|
}));
|
|
|
|
|
|
|
|
// right pane with preview canvas
|
|
|
|
m_canvas = new Bed_2D(this);
|
2019-07-18 10:56:52 +00:00
|
|
|
m_canvas->Bind(wxEVT_PAINT, [this](wxPaintEvent& e) { m_canvas->repaint(m_shape); });
|
|
|
|
m_canvas->Bind(wxEVT_SIZE, [this](wxSizeEvent& e) { m_canvas->Refresh(); });
|
2019-07-18 09:12:11 +00:00
|
|
|
|
|
|
|
wxSizer* left_sizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
left_sizer->Add(sbsizer, 0, wxEXPAND);
|
|
|
|
left_sizer->Add(texture_panel, 1, wxEXPAND);
|
2019-07-18 10:56:52 +00:00
|
|
|
left_sizer->Add(model_panel, 1, wxEXPAND);
|
2019-07-18 09:12:11 +00:00
|
|
|
|
|
|
|
wxSizer* top_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
top_sizer->Add(left_sizer, 0, wxEXPAND | wxLEFT | wxTOP | wxBOTTOM, 10);
|
|
|
|
top_sizer->Add(m_canvas, 1, wxEXPAND | wxALL, 10);
|
2018-01-25 12:46:04 +00:00
|
|
|
|
|
|
|
SetSizerAndFit(top_sizer);
|
|
|
|
|
|
|
|
set_shape(default_pt);
|
|
|
|
update_preview();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Called from the constructor.
|
|
|
|
// Create a panel for a rectangular / circular / custom bed shape.
|
2019-06-12 08:00:51 +00:00
|
|
|
ConfigOptionsGroupShp BedShapePanel::init_shape_options_page(const wxString& title)
|
2018-10-31 11:56:08 +00:00
|
|
|
{
|
2019-07-09 11:33:15 +00:00
|
|
|
wxPanel* panel = new wxPanel(m_shape_options_book);
|
|
|
|
ConfigOptionsGroupShp optgroup = std::make_shared<ConfigOptionsGroup>(panel, _(L("Settings")));
|
2018-01-25 12:46:04 +00:00
|
|
|
|
2019-04-18 00:03:40 +00:00
|
|
|
optgroup->label_width = 10;
|
2019-07-09 11:33:15 +00:00
|
|
|
optgroup->m_on_change = [this](t_config_option_key opt_key, boost::any value) {
|
|
|
|
update_shape();
|
|
|
|
};
|
|
|
|
|
|
|
|
m_optgroups.push_back(optgroup);
|
2020-09-25 08:44:22 +00:00
|
|
|
// panel->SetSizerAndFit(optgroup->sizer);
|
2019-07-09 11:33:15 +00:00
|
|
|
m_shape_options_book->AddPage(panel, title);
|
|
|
|
|
|
|
|
return optgroup;
|
2018-01-25 12:46:04 +00:00
|
|
|
}
|
|
|
|
|
2020-09-25 08:44:22 +00:00
|
|
|
void BedShapePanel::activate_options_page(ConfigOptionsGroupShp options_group)
|
|
|
|
{
|
|
|
|
options_group->activate();
|
|
|
|
options_group->parent()->SetSizerAndFit(options_group->sizer);
|
|
|
|
}
|
|
|
|
|
2019-07-18 09:12:11 +00:00
|
|
|
wxPanel* BedShapePanel::init_texture_panel()
|
|
|
|
{
|
|
|
|
wxPanel* panel = new wxPanel(this);
|
|
|
|
ConfigOptionsGroupShp optgroup = std::make_shared<ConfigOptionsGroup>(panel, _(L("Texture")));
|
|
|
|
|
|
|
|
optgroup->label_width = 10;
|
|
|
|
optgroup->m_on_change = [this](t_config_option_key opt_key, boost::any value) {
|
|
|
|
update_shape();
|
|
|
|
};
|
|
|
|
|
|
|
|
Line line{ "", "" };
|
|
|
|
line.full_width = 1;
|
|
|
|
line.widget = [this](wxWindow* parent) {
|
|
|
|
wxButton* load_btn = new wxButton(parent, wxID_ANY, _(L("Load...")));
|
|
|
|
wxSizer* load_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
load_sizer->Add(load_btn, 1, wxEXPAND);
|
|
|
|
|
|
|
|
wxStaticText* filename_lbl = new wxStaticText(parent, wxID_ANY, _(NONE));
|
|
|
|
wxSizer* filename_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
filename_sizer->Add(filename_lbl, 1, wxEXPAND);
|
|
|
|
|
|
|
|
wxButton* remove_btn = new wxButton(parent, wxID_ANY, _(L("Remove")));
|
|
|
|
wxSizer* remove_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
remove_sizer->Add(remove_btn, 1, wxEXPAND);
|
|
|
|
|
|
|
|
wxSizer* sizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
sizer->Add(filename_sizer, 1, wxEXPAND);
|
|
|
|
sizer->Add(load_sizer, 1, wxEXPAND);
|
|
|
|
sizer->Add(remove_sizer, 1, wxEXPAND);
|
|
|
|
|
|
|
|
load_btn->Bind(wxEVT_BUTTON, ([this](wxCommandEvent& e)
|
|
|
|
{
|
|
|
|
load_texture();
|
|
|
|
}));
|
|
|
|
|
|
|
|
remove_btn->Bind(wxEVT_BUTTON, ([this](wxCommandEvent& e)
|
|
|
|
{
|
|
|
|
m_custom_texture = NONE;
|
|
|
|
update_shape();
|
|
|
|
}));
|
|
|
|
|
|
|
|
filename_lbl->Bind(wxEVT_UPDATE_UI, ([this](wxUpdateUIEvent& e)
|
|
|
|
{
|
|
|
|
e.SetText(_(boost::filesystem::path(m_custom_texture).filename().string()));
|
2019-07-25 11:10:57 +00:00
|
|
|
wxStaticText* lbl = dynamic_cast<wxStaticText*>(e.GetEventObject());
|
|
|
|
if (lbl != nullptr)
|
|
|
|
{
|
2019-11-12 10:59:33 +00:00
|
|
|
bool exists = (m_custom_texture == NONE) || boost::filesystem::exists(m_custom_texture);
|
|
|
|
lbl->SetForegroundColour(exists ? wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT) : wxColor(*wxRED));
|
|
|
|
|
|
|
|
wxString tooltip_text = "";
|
|
|
|
if (m_custom_texture != NONE)
|
|
|
|
{
|
|
|
|
if (!exists)
|
2020-02-27 13:11:20 +00:00
|
|
|
tooltip_text += _(L("Not found:")) + " ";
|
2019-11-12 10:59:33 +00:00
|
|
|
|
|
|
|
tooltip_text += _(m_custom_texture);
|
|
|
|
}
|
|
|
|
|
2019-07-25 11:10:57 +00:00
|
|
|
wxToolTip* tooltip = lbl->GetToolTip();
|
|
|
|
if ((tooltip == nullptr) || (tooltip->GetTip() != tooltip_text))
|
|
|
|
lbl->SetToolTip(tooltip_text);
|
|
|
|
}
|
2019-07-18 09:12:11 +00:00
|
|
|
}));
|
|
|
|
|
|
|
|
remove_btn->Bind(wxEVT_UPDATE_UI, ([this](wxUpdateUIEvent& e)
|
|
|
|
{
|
|
|
|
e.Enable(m_custom_texture != NONE);
|
|
|
|
}));
|
|
|
|
|
|
|
|
return sizer;
|
|
|
|
};
|
|
|
|
optgroup->append_line(line);
|
2020-09-22 14:16:35 +00:00
|
|
|
optgroup->activate();
|
2019-07-18 09:12:11 +00:00
|
|
|
|
|
|
|
panel->SetSizerAndFit(optgroup->sizer);
|
|
|
|
|
|
|
|
return panel;
|
|
|
|
}
|
|
|
|
|
2019-07-18 10:56:52 +00:00
|
|
|
wxPanel* BedShapePanel::init_model_panel()
|
|
|
|
{
|
|
|
|
wxPanel* panel = new wxPanel(this);
|
|
|
|
ConfigOptionsGroupShp optgroup = std::make_shared<ConfigOptionsGroup>(panel, _(L("Model")));
|
|
|
|
|
|
|
|
optgroup->label_width = 10;
|
|
|
|
optgroup->m_on_change = [this](t_config_option_key opt_key, boost::any value) {
|
|
|
|
update_shape();
|
|
|
|
};
|
|
|
|
|
|
|
|
Line line{ "", "" };
|
|
|
|
line.full_width = 1;
|
|
|
|
line.widget = [this](wxWindow* parent) {
|
|
|
|
wxButton* load_btn = new wxButton(parent, wxID_ANY, _(L("Load...")));
|
|
|
|
wxSizer* load_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
load_sizer->Add(load_btn, 1, wxEXPAND);
|
|
|
|
|
|
|
|
wxStaticText* filename_lbl = new wxStaticText(parent, wxID_ANY, _(NONE));
|
|
|
|
wxSizer* filename_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
filename_sizer->Add(filename_lbl, 1, wxEXPAND);
|
|
|
|
|
|
|
|
wxButton* remove_btn = new wxButton(parent, wxID_ANY, _(L("Remove")));
|
|
|
|
wxSizer* remove_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
remove_sizer->Add(remove_btn, 1, wxEXPAND);
|
|
|
|
|
|
|
|
wxSizer* sizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
sizer->Add(filename_sizer, 1, wxEXPAND);
|
|
|
|
sizer->Add(load_sizer, 1, wxEXPAND);
|
|
|
|
sizer->Add(remove_sizer, 1, wxEXPAND);
|
|
|
|
|
|
|
|
load_btn->Bind(wxEVT_BUTTON, ([this](wxCommandEvent& e)
|
|
|
|
{
|
|
|
|
load_model();
|
|
|
|
}));
|
|
|
|
|
|
|
|
remove_btn->Bind(wxEVT_BUTTON, ([this](wxCommandEvent& e)
|
|
|
|
{
|
|
|
|
m_custom_model = NONE;
|
|
|
|
update_shape();
|
|
|
|
}));
|
|
|
|
|
|
|
|
filename_lbl->Bind(wxEVT_UPDATE_UI, ([this](wxUpdateUIEvent& e)
|
|
|
|
{
|
|
|
|
e.SetText(_(boost::filesystem::path(m_custom_model).filename().string()));
|
2019-07-25 11:10:57 +00:00
|
|
|
wxStaticText* lbl = dynamic_cast<wxStaticText*>(e.GetEventObject());
|
|
|
|
if (lbl != nullptr)
|
|
|
|
{
|
2019-11-12 10:59:33 +00:00
|
|
|
bool exists = (m_custom_model == NONE) || boost::filesystem::exists(m_custom_model);
|
|
|
|
lbl->SetForegroundColour(exists ? wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT) : wxColor(*wxRED));
|
|
|
|
|
|
|
|
wxString tooltip_text = "";
|
|
|
|
if (m_custom_model != NONE)
|
|
|
|
{
|
|
|
|
if (!exists)
|
2020-02-27 13:11:20 +00:00
|
|
|
tooltip_text += _(L("Not found:")) + " ";
|
2019-11-12 10:59:33 +00:00
|
|
|
|
|
|
|
tooltip_text += _(m_custom_model);
|
|
|
|
}
|
|
|
|
|
2019-07-25 11:10:57 +00:00
|
|
|
wxToolTip* tooltip = lbl->GetToolTip();
|
|
|
|
if ((tooltip == nullptr) || (tooltip->GetTip() != tooltip_text))
|
|
|
|
lbl->SetToolTip(tooltip_text);
|
|
|
|
}
|
2019-07-18 10:56:52 +00:00
|
|
|
}));
|
|
|
|
|
|
|
|
remove_btn->Bind(wxEVT_UPDATE_UI, ([this](wxUpdateUIEvent& e)
|
|
|
|
{
|
|
|
|
e.Enable(m_custom_model != NONE);
|
|
|
|
}));
|
|
|
|
|
|
|
|
return sizer;
|
|
|
|
};
|
|
|
|
optgroup->append_line(line);
|
2020-09-22 14:16:35 +00:00
|
|
|
optgroup->activate();
|
2019-07-18 10:56:52 +00:00
|
|
|
|
|
|
|
panel->SetSizerAndFit(optgroup->sizer);
|
|
|
|
|
|
|
|
return panel;
|
|
|
|
}
|
|
|
|
|
2018-01-25 12:46:04 +00:00
|
|
|
// Called from the constructor.
|
|
|
|
// Set the initial bed shape from a list of points.
|
|
|
|
// Deduce the bed shape type(rect, circle, custom)
|
|
|
|
// This routine shall be smart enough if the user messes up
|
|
|
|
// with the list of points in the ini file directly.
|
2019-07-17 12:53:02 +00:00
|
|
|
void BedShapePanel::set_shape(const ConfigOptionPoints& points)
|
2018-01-25 12:46:04 +00:00
|
|
|
{
|
2020-08-19 13:35:58 +00:00
|
|
|
BedShape shape(points);
|
2018-03-23 16:27:43 +00:00
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
m_shape_options_book->SetSelection(shape.get_type());
|
|
|
|
shape.apply_optgroup_values(m_optgroups[shape.get_type()]);
|
2018-01-25 12:46:04 +00:00
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
// Copy the polygon to the canvas, make a copy of the array, if custom shape is selected
|
|
|
|
if (shape.is_custom())
|
|
|
|
m_loaded_shape = points.values;
|
2018-01-25 12:46:04 +00:00
|
|
|
|
2019-07-02 13:10:59 +00:00
|
|
|
update_shape();
|
2020-08-19 13:35:58 +00:00
|
|
|
|
|
|
|
return;
|
2018-01-25 12:46:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void BedShapePanel::update_preview()
|
|
|
|
{
|
2018-01-30 11:10:12 +00:00
|
|
|
if (m_canvas) m_canvas->Refresh();
|
2018-01-25 12:46:04 +00:00
|
|
|
Refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the bed shape from the dialog fields.
|
|
|
|
void BedShapePanel::update_shape()
|
|
|
|
{
|
|
|
|
auto page_idx = m_shape_options_book->GetSelection();
|
2020-08-19 13:35:58 +00:00
|
|
|
auto opt_group = m_optgroups[page_idx];
|
|
|
|
|
|
|
|
BedShape::Type page_type = static_cast<BedShape::Type>(page_idx);
|
|
|
|
|
|
|
|
if (page_type == BedShape::Type::Rectangular) {
|
2018-08-21 19:05:24 +00:00
|
|
|
Vec2d rect_size(Vec2d::Zero());
|
|
|
|
Vec2d rect_origin(Vec2d::Zero());
|
2020-08-19 13:35:58 +00:00
|
|
|
|
|
|
|
try { rect_size = boost::any_cast<Vec2d>(opt_group->get_value("rect_size")); }
|
|
|
|
catch (const std::exception& /* e */) { return; }
|
|
|
|
|
|
|
|
try { rect_origin = boost::any_cast<Vec2d>(opt_group->get_value("rect_origin")); }
|
|
|
|
catch (const std::exception & /* e */) { return; }
|
|
|
|
|
2018-08-17 13:53:43 +00:00
|
|
|
auto x = rect_size(0);
|
|
|
|
auto y = rect_size(1);
|
2018-01-25 12:46:04 +00:00
|
|
|
// empty strings or '-' or other things
|
2018-01-30 11:10:12 +00:00
|
|
|
if (x == 0 || y == 0) return;
|
|
|
|
double x0 = 0.0;
|
|
|
|
double y0 = 0.0;
|
|
|
|
double x1 = x;
|
|
|
|
double y1 = y;
|
|
|
|
|
2018-08-17 13:53:43 +00:00
|
|
|
auto dx = rect_origin(0);
|
|
|
|
auto dy = rect_origin(1);
|
2018-01-30 11:10:12 +00:00
|
|
|
|
|
|
|
x0 -= dx;
|
|
|
|
x1 -= dx;
|
|
|
|
y0 -= dy;
|
|
|
|
y1 -= dy;
|
2019-07-18 09:12:11 +00:00
|
|
|
m_shape = { Vec2d(x0, y0),
|
|
|
|
Vec2d(x1, y0),
|
|
|
|
Vec2d(x1, y1),
|
|
|
|
Vec2d(x0, y1) };
|
|
|
|
}
|
2020-08-19 13:35:58 +00:00
|
|
|
else if (page_type == BedShape::Type::Circular) {
|
2018-01-30 11:10:12 +00:00
|
|
|
double diameter;
|
2020-08-19 13:35:58 +00:00
|
|
|
try { diameter = boost::any_cast<double>(opt_group->get_value("diameter")); }
|
|
|
|
catch (const std::exception & /* e */) { return; }
|
|
|
|
|
2018-01-30 11:10:12 +00:00
|
|
|
if (diameter == 0.0) return ;
|
|
|
|
auto r = diameter / 2;
|
|
|
|
auto twopi = 2 * PI;
|
2019-07-18 09:12:11 +00:00
|
|
|
auto edges = 72;
|
|
|
|
std::vector<Vec2d> points;
|
2019-09-04 08:46:51 +00:00
|
|
|
for (int i = 1; i <= edges; ++i) {
|
2019-07-18 09:12:11 +00:00
|
|
|
auto angle = i * twopi / edges;
|
2018-08-21 19:05:24 +00:00
|
|
|
points.push_back(Vec2d(r*cos(angle), r*sin(angle)));
|
2018-01-30 11:10:12 +00:00
|
|
|
}
|
2019-07-18 09:12:11 +00:00
|
|
|
m_shape = points;
|
|
|
|
}
|
2020-08-19 13:35:58 +00:00
|
|
|
else if (page_type == BedShape::Type::Custom)
|
2019-07-18 09:12:11 +00:00
|
|
|
m_shape = m_loaded_shape;
|
2018-01-25 12:46:04 +00:00
|
|
|
|
2019-09-24 14:01:01 +00:00
|
|
|
update_preview();
|
2018-01-25 12:46:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Loads an stl file, projects it to the XY plane and calculates a polygon.
|
|
|
|
void BedShapePanel::load_stl()
|
|
|
|
{
|
2019-07-18 09:12:11 +00:00
|
|
|
wxFileDialog dialog(this, _(L("Choose an STL file to import bed shape from:")), "", "", file_wildcards(FT_STL), wxFD_OPEN | wxFD_FILE_MUST_EXIST);
|
2019-07-09 11:33:15 +00:00
|
|
|
if (dialog.ShowModal() != wxID_OK)
|
|
|
|
return;
|
|
|
|
|
2020-08-19 13:35:58 +00:00
|
|
|
std::string file_name = dialog.GetPath().ToUTF8().data();
|
|
|
|
if (!boost::algorithm::iends_with(file_name, ".stl"))
|
2019-07-18 09:12:11 +00:00
|
|
|
{
|
|
|
|
show_error(this, _(L("Invalid file format.")));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxBusyCursor wait;
|
2018-01-30 11:10:12 +00:00
|
|
|
|
|
|
|
Model model;
|
|
|
|
try {
|
2020-08-19 13:35:58 +00:00
|
|
|
model = Model::read_from_file(file_name);
|
2018-01-30 11:10:12 +00:00
|
|
|
}
|
2019-07-17 12:53:02 +00:00
|
|
|
catch (std::exception &) {
|
2019-07-09 11:33:15 +00:00
|
|
|
show_error(this, _(L("Error! Invalid model")));
|
|
|
|
return;
|
|
|
|
}
|
2018-01-30 11:10:12 +00:00
|
|
|
|
|
|
|
auto mesh = model.mesh();
|
|
|
|
auto expolygons = mesh.horizontal_projection();
|
|
|
|
|
|
|
|
if (expolygons.size() == 0) {
|
2018-02-23 08:16:35 +00:00
|
|
|
show_error(this, _(L("The selected file contains no geometry.")));
|
2018-01-30 11:10:12 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (expolygons.size() > 1) {
|
2018-02-23 08:16:35 +00:00
|
|
|
show_error(this, _(L("The selected file contains several disjoint areas. This is not supported.")));
|
2018-01-30 11:10:12 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto polygon = expolygons[0].contour;
|
2018-08-21 19:05:24 +00:00
|
|
|
std::vector<Vec2d> points;
|
2018-01-30 11:10:12 +00:00
|
|
|
for (auto pt : polygon.points)
|
2018-08-21 15:43:05 +00:00
|
|
|
points.push_back(unscale(pt));
|
2019-06-12 08:00:51 +00:00
|
|
|
|
2019-07-18 09:12:11 +00:00
|
|
|
m_loaded_shape = points;
|
|
|
|
update_shape();
|
|
|
|
}
|
|
|
|
|
|
|
|
void BedShapePanel::load_texture()
|
|
|
|
{
|
|
|
|
wxFileDialog dialog(this, _(L("Choose a file to import bed texture from (PNG/SVG):")), "", "",
|
|
|
|
file_wildcards(FT_TEX), wxFD_OPEN | wxFD_FILE_MUST_EXIST);
|
|
|
|
|
|
|
|
if (dialog.ShowModal() != wxID_OK)
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_custom_texture = NONE;
|
|
|
|
|
|
|
|
std::string file_name = dialog.GetPath().ToUTF8().data();
|
2019-07-18 13:03:05 +00:00
|
|
|
if (!boost::algorithm::iends_with(file_name, ".png") && !boost::algorithm::iends_with(file_name, ".svg"))
|
2019-07-18 09:12:11 +00:00
|
|
|
{
|
|
|
|
show_error(this, _(L("Invalid file format.")));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxBusyCursor wait;
|
|
|
|
|
|
|
|
m_custom_texture = file_name;
|
2019-06-12 08:00:51 +00:00
|
|
|
update_shape();
|
2018-01-25 12:46:04 +00:00
|
|
|
}
|
|
|
|
|
2019-07-18 10:56:52 +00:00
|
|
|
void BedShapePanel::load_model()
|
|
|
|
{
|
|
|
|
wxFileDialog dialog(this, _(L("Choose an STL file to import bed model from:")), "", "",
|
|
|
|
file_wildcards(FT_STL), wxFD_OPEN | wxFD_FILE_MUST_EXIST);
|
|
|
|
|
|
|
|
if (dialog.ShowModal() != wxID_OK)
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_custom_model = NONE;
|
|
|
|
|
|
|
|
std::string file_name = dialog.GetPath().ToUTF8().data();
|
2019-07-18 13:03:05 +00:00
|
|
|
if (!boost::algorithm::iends_with(file_name, ".stl"))
|
2019-07-18 10:56:52 +00:00
|
|
|
{
|
|
|
|
show_error(this, _(L("Invalid file format.")));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
wxBusyCursor wait;
|
|
|
|
|
|
|
|
m_custom_model = file_name;
|
|
|
|
update_shape();
|
|
|
|
}
|
|
|
|
|
2018-01-25 12:46:04 +00:00
|
|
|
} // GUI
|
|
|
|
} // Slic3r
|