mirror of
https://git.mirrors.martin98.com/https://github.com/prusa3d/PrusaSlicer.git
synced 2025-05-08 14:59:09 +08:00

Added some functions to TabIface. Added BedShapeDialog & Bed2D classes. Added new_scale to Polygon. In class Field: Point renamed to PointCtrl and added set_value for PointCtrl, extended get_value for Choice.
315 lines
9.7 KiB
C++
315 lines
9.7 KiB
C++
#include "BedShapeDialog.hpp"
|
||
|
||
#include <wx/sizer.h>
|
||
#include <wx/statbox.h>
|
||
#include <wx/wx.h>
|
||
#include "Polygon.hpp"
|
||
#include "BoundingBox.hpp"
|
||
#include <wx/numformatter.h>
|
||
|
||
namespace Slic3r {
|
||
namespace GUI {
|
||
|
||
void BedShapeDialog::build_dialog(ConfigOptionPoints* default_pt)
|
||
{
|
||
m_panel = new BedShapePanel(this);
|
||
m_panel->build_panel(default_pt);
|
||
|
||
auto main_sizer = new wxBoxSizer(wxVERTICAL);
|
||
main_sizer->Add(m_panel, 1, wxEXPAND);
|
||
main_sizer->Add(CreateButtonSizer(wxOK | wxCANCEL), 0, /*wxEXPAND*/wxALIGN_CENTER_HORIZONTAL | wxBOTTOM/*ALL*/, 10);
|
||
|
||
SetSizer(main_sizer);
|
||
SetMinSize(GetSize());
|
||
main_sizer->SetSizeHints(this);
|
||
|
||
// needed to actually free memory
|
||
this->Bind(wxEVT_CLOSE_WINDOW, ([this](wxCloseEvent e){
|
||
EndModal(wxID_OK);
|
||
Destroy();
|
||
}));
|
||
}
|
||
|
||
void BedShapePanel::build_panel(ConfigOptionPoints* default_pt)
|
||
{
|
||
// on_change(nullptr);
|
||
|
||
auto box = new wxStaticBox(this, wxID_ANY, "Shape");
|
||
auto sbsizer = new wxStaticBoxSizer(box, wxVERTICAL);
|
||
|
||
// shape options
|
||
m_shape_options_book = new wxChoicebook(this, wxID_ANY, wxDefaultPosition, wxSize(300, -1), wxCHB_TOP);
|
||
sbsizer->Add(m_shape_options_book);
|
||
|
||
auto optgroup = init_shape_options_page("Rectangular");
|
||
ConfigOptionDef def;
|
||
def.type = coPoints;
|
||
def.default_value = new ConfigOptionPoints{ Pointf(200, 200) };
|
||
def.label = "Size";
|
||
def.tooltip = "Size in X and Y of the rectangular plate.";
|
||
Option option(def, "rect_size");
|
||
optgroup->append_single_option_line(option);
|
||
|
||
def.type = coPoints;
|
||
def.default_value = new ConfigOptionPoints{ Pointf(0, 0) };
|
||
def.label = "Origin";
|
||
def.tooltip = "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);
|
||
|
||
optgroup = init_shape_options_page("Circular");
|
||
def.type = coFloat;
|
||
def.default_value = new ConfigOptionFloat(200);
|
||
def.sidetext = "mm";
|
||
def.label = "Diameter";
|
||
def.tooltip = "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);
|
||
|
||
optgroup = init_shape_options_page("Custom");
|
||
Line line{ "", "" };
|
||
line.full_width = 1;
|
||
line.widget = [this](wxWindow* parent) {
|
||
auto btn = new wxButton(parent, wxID_ANY, "Load shape from STL...", wxDefaultPosition, wxDefaultSize);
|
||
|
||
auto sizer = new wxBoxSizer(wxHORIZONTAL);
|
||
sizer->Add(btn);
|
||
|
||
btn->Bind(wxEVT_BUTTON, ([this](wxCommandEvent e)
|
||
{
|
||
load_stl();
|
||
}));
|
||
|
||
return sizer;
|
||
};
|
||
optgroup->append_line(line);
|
||
|
||
Bind(wxEVT_CHOICEBOOK_PAGE_CHANGED, ([this](wxCommandEvent e)
|
||
{
|
||
update_shape();
|
||
}));
|
||
|
||
// right pane with preview canvas
|
||
m_canvas = new Bed_2D(this);
|
||
m_canvas->m_bed_shape = default_pt->values;
|
||
|
||
// main sizer
|
||
auto top_sizer = new wxBoxSizer(wxHORIZONTAL);
|
||
top_sizer->Add(sbsizer, 0, wxEXPAND | wxLeft | wxTOP | wxBOTTOM, 10);
|
||
if (m_canvas)
|
||
top_sizer->Add(m_canvas, 1, wxEXPAND | wxALL, 10) ;
|
||
|
||
SetSizerAndFit(top_sizer);
|
||
|
||
set_shape(default_pt);
|
||
update_preview();
|
||
}
|
||
|
||
#define SHAPE_RECTANGULAR 0
|
||
#define SHAPE_CIRCULAR 1
|
||
#define SHAPE_CUSTOM 2
|
||
|
||
// Called from the constructor.
|
||
// Create a panel for a rectangular / circular / custom bed shape.
|
||
ConfigOptionsGroupShp BedShapePanel::init_shape_options_page(std::string title){
|
||
|
||
auto panel = new wxPanel(m_shape_options_book);
|
||
ConfigOptionsGroupShp optgroup;
|
||
optgroup = std::make_shared<ConfigOptionsGroup>(panel, "Settings");
|
||
|
||
optgroup->label_width = 100;
|
||
optgroup->m_on_change = [this](t_config_option_key opt_key, boost::any value){
|
||
update_shape();
|
||
};
|
||
|
||
m_optgroups.push_back(optgroup);
|
||
panel->SetSizerAndFit(optgroup->sizer);
|
||
m_shape_options_book->AddPage(panel, title);
|
||
|
||
return optgroup;
|
||
}
|
||
|
||
// 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.
|
||
void BedShapePanel::set_shape(ConfigOptionPoints* points)
|
||
{
|
||
auto polygon = Polygon::new_scale(points->values);
|
||
|
||
// 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
|
||
// the || 0 hack prevents "-0" which might confuse the user
|
||
int x_min, x_max, y_min, y_max;
|
||
x_max = x_min = points->values[0].x;
|
||
y_max = y_min = points->values[0].y;
|
||
for (auto pt : points->values){
|
||
if (x_min > pt.x) x_min = pt.x;
|
||
if (x_max < pt.x) x_max = pt.x;
|
||
if (y_min > pt.y) y_min = pt.y;
|
||
if (y_max < pt.y) y_max = pt.y;
|
||
}
|
||
if (x_min < 0) x_min = 0;
|
||
if (x_max < 0) x_max = 0;
|
||
if (y_min < 0) y_min = 0;
|
||
if (y_max < 0) y_max = 0;
|
||
// auto x_min = min(map $_->[X], @$points) || 0;
|
||
// auto x_max = max(map $_->[X], @$points) || 0;
|
||
// auto y_min = min(map $_->[Y], @$points) || 0;
|
||
// auto y_max = max(map $_->[Y], @$points) || 0;
|
||
// auto origin = [-x_min, -y_min];
|
||
auto origin = new ConfigOptionPoints{ Pointf(-x_min, -y_min) };
|
||
|
||
m_shape_options_book->SetSelection(SHAPE_RECTANGULAR);
|
||
auto optgroup = m_optgroups[SHAPE_RECTANGULAR];
|
||
optgroup->set_value("rect_size", new ConfigOptionPoints{ Pointf(x_max - x_min, y_max - y_min) });//[x_max - x_min, y_max - y_min]);
|
||
optgroup->set_value("rect_origin", origin);
|
||
update_shape();
|
||
return;
|
||
}
|
||
}
|
||
|
||
// is this a circle ?
|
||
{
|
||
// Analyze the array of points.Do they reside on a circle ?
|
||
// auto polygon;// = Slic3r::Polygon->new_scale(@$points);
|
||
auto center = polygon.bounding_box().center();// ->bounding_box->center;
|
||
// auto /*@*/vertex_distances;// = map $center->distance_to($_), @$polygon;
|
||
// auto avg_dist = sum(/*@*/vertex_distances) / /*@*/vertex_distances;
|
||
std::vector<double> vertex_distances;// = map $center->distance_to($_), @$polygon;
|
||
double avg_dist = 0;
|
||
for (auto pt: polygon.points)
|
||
{
|
||
double distance = center.distance_to(pt);
|
||
vertex_distances.push_back(distance);
|
||
avg_dist += distance;
|
||
}
|
||
|
||
bool defined_value = true;
|
||
for (auto el: vertex_distances)
|
||
{
|
||
if (abs(el - avg_dist) > 10 * SCALED_EPSILON/*scaled_epsilon*/)
|
||
defined_value = false;
|
||
break;
|
||
}
|
||
if (defined_value/*!defined first{ abs($_ - $avg_dist) > 10 * scaled_epsilon } @vertex_distances*/) {
|
||
// all vertices are equidistant to center
|
||
m_shape_options_book->SetSelection(SHAPE_CIRCULAR);
|
||
auto optgroup = m_optgroups[SHAPE_CIRCULAR];
|
||
boost::any ret = wxNumberFormatter::ToString(unscale(avg_dist * 2), 0);
|
||
optgroup->set_value("diameter", ret /*sprintf("%.0f", unscale(avg_dist * 2))*/);
|
||
update_shape();
|
||
return;
|
||
}
|
||
}
|
||
|
||
if (points->size() < 3) {
|
||
// Invalid polygon.Revert to default bed dimensions.
|
||
m_shape_options_book->SetSelection(SHAPE_RECTANGULAR);
|
||
auto optgroup = m_optgroups[SHAPE_RECTANGULAR];
|
||
optgroup->set_value("rect_size", new ConfigOptionPoints{ Pointf(200, 200) });
|
||
optgroup->set_value("rect_origin", new ConfigOptionPoints{ Pointf(0, 0) });
|
||
update_shape();
|
||
return;
|
||
}
|
||
|
||
// This is a custom bed shape, use the polygon provided.
|
||
m_shape_options_book->SetSelection(SHAPE_CUSTOM);
|
||
// Copy the polygon to the canvas, make a copy of the array.
|
||
m_canvas->m_bed_shape = points->values;
|
||
update_shape();
|
||
}
|
||
|
||
void BedShapePanel::update_preview()
|
||
{
|
||
if (m_canvas) m_canvas->Refresh();
|
||
Refresh();
|
||
}
|
||
|
||
// Update the bed shape from the dialog fields.
|
||
void BedShapePanel::update_shape()
|
||
{
|
||
auto page_idx = m_shape_options_book->GetSelection();
|
||
if (page_idx == SHAPE_RECTANGULAR) {
|
||
auto rect_size = m_optgroups[SHAPE_RECTANGULAR]->get_value("rect_size");
|
||
auto rect_origin = m_optgroups[SHAPE_RECTANGULAR]->get_value("rect_origin");
|
||
/* auto x = rect_size->x;
|
||
auto y = rect_size->y;
|
||
// empty strings or '-' or other things
|
||
// if (!looks_like_number(x) || !looks_like_number(y)) return;
|
||
if ((!x || !y) || x == 0 || y == 0) return;
|
||
// my($x0, $y0) = (0, 0);
|
||
// my($x1, $y1) = ($x, $y);
|
||
// {
|
||
// my($dx, $dy) = @$rect_origin;
|
||
// return if !looks_like_number($dx) || !looks_like_number($dy); #<23>empty strings or '-' or other things
|
||
// $x0 -= $dx;
|
||
// $x1 -= $dx;
|
||
// $y0 -= $dy;
|
||
// $y1 -= $dy;
|
||
// }
|
||
// m_canvas->bed_shape([
|
||
// [$x0, $y0],
|
||
// [$x1, $y0],
|
||
// [$x1, $y1],
|
||
// [$x0, $y1],
|
||
// ]);
|
||
}
|
||
else if(page_idx == SHAPE_CIRCULAR) {
|
||
// auto diameter = m_optgroups[SHAPE_CIRCULAR]->get_value("diameter");
|
||
// if (!diameter || diameter == 0) return ;
|
||
// r = diameter / 2;
|
||
// twopi = 2 * PI;
|
||
// edges = 60;
|
||
// polygon = Slic3r::Polygon->new_scale(
|
||
// map[$r * cos $_, $r * sin $_],
|
||
// map{ $twopi / $edges*$_ } 1..$edges
|
||
// );
|
||
// m_canvas->bed_shape([
|
||
// map[unscale($_->x), unscale($_->y)], @$polygon #))
|
||
// ]);
|
||
*/ }
|
||
|
||
// $self->{on_change}->();
|
||
update_preview();
|
||
}
|
||
|
||
// Loads an stl file, projects it to the XY plane and calculates a polygon.
|
||
void BedShapePanel::load_stl()
|
||
{
|
||
auto dialog = new wxFileDialog(this, "Choose a file to import bed shape from (STL/OBJ/AMF/PRUSA):", "", "",
|
||
wxFileSelectorDefaultWildcardStr/* &Slic3r::GUI::MODEL_WILDCARD*/, wxFD_OPEN | wxFD_FILE_MUST_EXIST);
|
||
if (dialog->ShowModal() != wxID_OK) {
|
||
dialog->Destroy();
|
||
return;
|
||
}
|
||
wxArrayString input_file;
|
||
dialog->GetPaths(input_file);
|
||
dialog->Destroy();
|
||
|
||
// auto model = Slic3r::Model->read_from_file(input_file);
|
||
// auto mesh = model->mesh;
|
||
// auto expolygons = mesh->horizontal_projection;
|
||
//
|
||
// if (expolygons.size() == 0) {
|
||
// show_error(this, "The selected file contains no geometry.");
|
||
// return;
|
||
// }
|
||
// if (expolygons.size() > 1) {
|
||
// show_error(this, "The selected file contains several disjoint areas. This is not supported.");
|
||
// return;
|
||
// }
|
||
//
|
||
// auto polygon = expolygons[0]->contour;
|
||
// m_canvas->bed_shape([map[unscale($_->x), unscale($_->y)], @$polygon]);
|
||
update_preview();
|
||
}
|
||
|
||
} // GUI
|
||
} // Slic3r
|