2018-11-22 14:29:59 +00:00
|
|
|
#include "libslic3r/libslic3r.h"
|
2018-09-18 11:35:05 +00:00
|
|
|
#include "GUI_Preview.hpp"
|
2018-11-07 13:44:47 +00:00
|
|
|
#include "GUI_App.hpp"
|
2018-09-18 11:35:05 +00:00
|
|
|
#include "GUI.hpp"
|
2018-11-26 13:41:58 +00:00
|
|
|
#include "I18N.hpp"
|
2018-09-18 11:35:05 +00:00
|
|
|
#include "3DScene.hpp"
|
2018-11-22 14:29:59 +00:00
|
|
|
#include "BackgroundSlicingProcess.hpp"
|
2020-05-06 12:38:53 +00:00
|
|
|
#include "OpenGLManager.hpp"
|
2018-11-23 11:47:32 +00:00
|
|
|
#include "GLCanvas3D.hpp"
|
2020-06-16 14:58:41 +00:00
|
|
|
#include "libslic3r/PresetBundle.hpp"
|
2020-01-23 14:07:31 +00:00
|
|
|
#include "DoubleSlider.hpp"
|
2020-05-26 15:42:57 +00:00
|
|
|
#include "Plater.hpp"
|
2020-06-05 14:50:17 +00:00
|
|
|
#include "MainFrame.hpp"
|
2018-09-18 11:35:05 +00:00
|
|
|
|
|
|
|
#include <wx/notebook.h>
|
|
|
|
#include <wx/glcanvas.h>
|
|
|
|
#include <wx/sizer.h>
|
|
|
|
#include <wx/stattext.h>
|
|
|
|
#include <wx/choice.h>
|
|
|
|
#include <wx/combo.h>
|
|
|
|
#include <wx/checkbox.h>
|
|
|
|
|
2018-09-18 13:50:52 +00:00
|
|
|
// this include must follow the wxWidgets ones or it won't compile on Windows -> see http://trac.wxwidgets.org/ticket/2421
|
2018-11-26 13:41:58 +00:00
|
|
|
#include "libslic3r/Print.hpp"
|
2018-11-26 15:17:59 +00:00
|
|
|
#include "libslic3r/SLAPrint.hpp"
|
2018-09-18 13:50:52 +00:00
|
|
|
|
2018-09-18 11:35:05 +00:00
|
|
|
namespace Slic3r {
|
|
|
|
namespace GUI {
|
|
|
|
|
2020-03-02 09:58:46 +00:00
|
|
|
View3D::View3D(wxWindow* parent, Model* model, DynamicPrintConfig* config, BackgroundSlicingProcess* process)
|
|
|
|
: m_canvas_widget(nullptr)
|
|
|
|
, m_canvas(nullptr)
|
|
|
|
{
|
|
|
|
init(parent, model, config, process);
|
|
|
|
}
|
2018-12-04 12:55:25 +00:00
|
|
|
|
|
|
|
View3D::~View3D()
|
|
|
|
{
|
2021-01-18 08:33:19 +00:00
|
|
|
delete m_canvas;
|
|
|
|
delete m_canvas_widget;
|
2018-12-04 12:55:25 +00:00
|
|
|
}
|
|
|
|
|
2020-03-02 09:58:46 +00:00
|
|
|
bool View3D::init(wxWindow* parent, Model* model, DynamicPrintConfig* config, BackgroundSlicingProcess* process)
|
2018-12-04 12:55:25 +00:00
|
|
|
{
|
2019-02-21 17:31:01 +00:00
|
|
|
if (!Create(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0 /* disable wxTAB_TRAVERSAL */))
|
2018-12-04 12:55:25 +00:00
|
|
|
return false;
|
|
|
|
|
2020-05-06 12:38:53 +00:00
|
|
|
m_canvas_widget = OpenGLManager::create_wxglcanvas(*this);
|
2020-03-02 09:58:46 +00:00
|
|
|
if (m_canvas_widget == nullptr)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
m_canvas = new GLCanvas3D(m_canvas_widget);
|
|
|
|
m_canvas->set_context(wxGetApp().init_glcontext(*m_canvas_widget));
|
2018-12-04 12:55:25 +00:00
|
|
|
|
2020-05-06 12:38:53 +00:00
|
|
|
m_canvas->allow_multisample(OpenGLManager::can_multisample());
|
2018-12-04 12:55:25 +00:00
|
|
|
// XXX: If have OpenGL
|
|
|
|
m_canvas->enable_picking(true);
|
|
|
|
m_canvas->enable_moving(true);
|
|
|
|
// XXX: more config from 3D.pm
|
|
|
|
m_canvas->set_model(model);
|
|
|
|
m_canvas->set_process(process);
|
|
|
|
m_canvas->set_config(config);
|
|
|
|
m_canvas->enable_gizmos(true);
|
2019-03-28 07:44:46 +00:00
|
|
|
m_canvas->enable_selection(true);
|
2019-07-19 13:36:55 +00:00
|
|
|
m_canvas->enable_main_toolbar(true);
|
|
|
|
m_canvas->enable_undoredo_toolbar(true);
|
2020-01-28 12:15:21 +00:00
|
|
|
m_canvas->enable_labels(true);
|
2020-03-13 14:09:07 +00:00
|
|
|
m_canvas->enable_slope(true);
|
2018-12-04 12:55:25 +00:00
|
|
|
|
|
|
|
wxBoxSizer* main_sizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
main_sizer->Add(m_canvas_widget, 1, wxALL | wxEXPAND, 0);
|
|
|
|
|
|
|
|
SetSizer(main_sizer);
|
|
|
|
SetMinSize(GetSize());
|
|
|
|
GetSizer()->SetSizeHints(this);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2018-11-07 13:44:47 +00:00
|
|
|
|
2018-12-04 12:55:25 +00:00
|
|
|
void View3D::set_as_dirty()
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->set_as_dirty();
|
|
|
|
}
|
|
|
|
|
2019-02-19 14:15:27 +00:00
|
|
|
void View3D::bed_shape_changed()
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->bed_shape_changed();
|
|
|
|
}
|
2018-12-04 12:55:25 +00:00
|
|
|
|
|
|
|
void View3D::select_view(const std::string& direction)
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->select_view(direction);
|
|
|
|
}
|
|
|
|
|
|
|
|
void View3D::select_all()
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->select_all();
|
|
|
|
}
|
|
|
|
|
2019-05-14 09:57:39 +00:00
|
|
|
void View3D::deselect_all()
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->deselect_all();
|
|
|
|
}
|
|
|
|
|
2018-12-04 12:55:25 +00:00
|
|
|
void View3D::delete_selected()
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->delete_selected();
|
|
|
|
}
|
|
|
|
|
|
|
|
void View3D::mirror_selection(Axis axis)
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->mirror_selection(axis);
|
|
|
|
}
|
|
|
|
|
|
|
|
int View3D::check_volumes_outside_state() const
|
|
|
|
{
|
|
|
|
return (m_canvas != nullptr) ? m_canvas->check_volumes_outside_state() : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool View3D::is_layers_editing_enabled() const
|
|
|
|
{
|
|
|
|
return (m_canvas != nullptr) ? m_canvas->is_layers_editing_enabled() : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool View3D::is_layers_editing_allowed() const
|
|
|
|
{
|
|
|
|
return (m_canvas != nullptr) ? m_canvas->is_layers_editing_allowed() : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void View3D::enable_layers_editing(bool enable)
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->enable_layers_editing(enable);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool View3D::is_dragging() const
|
|
|
|
{
|
|
|
|
return (m_canvas != nullptr) ? m_canvas->is_dragging() : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool View3D::is_reload_delayed() const
|
|
|
|
{
|
|
|
|
return (m_canvas != nullptr) ? m_canvas->is_reload_delayed() : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void View3D::reload_scene(bool refresh_immediately, bool force_full_scene_refresh)
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->reload_scene(refresh_immediately, force_full_scene_refresh);
|
|
|
|
}
|
|
|
|
|
|
|
|
void View3D::render()
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
2019-03-04 08:41:25 +00:00
|
|
|
//m_canvas->render();
|
|
|
|
m_canvas->set_as_dirty();
|
2018-12-04 12:55:25 +00:00
|
|
|
}
|
|
|
|
|
2020-03-30 07:36:37 +00:00
|
|
|
Preview::Preview(
|
2020-05-07 08:49:12 +00:00
|
|
|
wxWindow* parent, Model* model, DynamicPrintConfig* config,
|
|
|
|
BackgroundSlicingProcess* process, GCodeProcessor::Result* gcode_result, std::function<void()> schedule_background_process_func)
|
2020-12-02 10:38:10 +00:00
|
|
|
: m_config(config)
|
2018-11-22 14:29:59 +00:00
|
|
|
, m_process(process)
|
2020-03-02 14:13:23 +00:00
|
|
|
, m_gcode_result(gcode_result)
|
2020-12-04 11:31:50 +00:00
|
|
|
#if !ENABLE_PREVIEW_TYPE_CHANGE
|
2018-09-18 11:35:05 +00:00
|
|
|
, m_preferred_color_mode("feature")
|
2020-12-04 11:31:50 +00:00
|
|
|
#endif // !ENABLE_PREVIEW_TYPE_CHANGE
|
2018-11-12 14:35:58 +00:00
|
|
|
, m_schedule_background_process(schedule_background_process_func)
|
2018-09-18 11:35:05 +00:00
|
|
|
{
|
2020-11-19 11:29:19 +00:00
|
|
|
if (init(parent, model))
|
2018-12-04 12:55:25 +00:00
|
|
|
load_print();
|
2018-09-18 11:35:05 +00:00
|
|
|
}
|
|
|
|
|
2020-03-02 09:58:46 +00:00
|
|
|
bool Preview::init(wxWindow* parent, Model* model)
|
2018-09-18 11:35:05 +00:00
|
|
|
{
|
2019-02-21 10:54:18 +00:00
|
|
|
if (!Create(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0 /* disable wxTAB_TRAVERSAL */))
|
2018-12-04 12:55:25 +00:00
|
|
|
return false;
|
2018-09-18 11:35:05 +00:00
|
|
|
|
2020-08-19 11:14:47 +00:00
|
|
|
// to match the background of the sliders
|
|
|
|
#ifdef _WIN32
|
2020-08-19 10:59:50 +00:00
|
|
|
SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW));
|
2020-08-19 11:14:47 +00:00
|
|
|
#else
|
|
|
|
SetBackgroundColour(GetParent()->GetBackgroundColour());
|
|
|
|
#endif // _WIN32
|
2020-08-19 10:59:50 +00:00
|
|
|
|
2020-05-06 12:38:53 +00:00
|
|
|
m_canvas_widget = OpenGLManager::create_wxglcanvas(*this);
|
2020-03-02 09:58:46 +00:00
|
|
|
if (m_canvas_widget == nullptr)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
m_canvas = new GLCanvas3D(m_canvas_widget);
|
|
|
|
m_canvas->set_context(wxGetApp().init_glcontext(*m_canvas_widget));
|
2020-05-06 12:38:53 +00:00
|
|
|
m_canvas->allow_multisample(OpenGLManager::can_multisample());
|
2018-11-23 11:47:32 +00:00
|
|
|
m_canvas->set_config(m_config);
|
2019-04-02 11:47:49 +00:00
|
|
|
m_canvas->set_model(model);
|
2019-03-13 13:35:09 +00:00
|
|
|
m_canvas->set_process(m_process);
|
2018-11-23 11:47:32 +00:00
|
|
|
m_canvas->enable_legend_texture(true);
|
|
|
|
m_canvas->enable_dynamic_background(true);
|
2018-09-18 11:35:05 +00:00
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider_sizer = create_layers_slider_sizer();
|
2020-05-19 08:04:14 +00:00
|
|
|
|
|
|
|
m_bottom_toolbar_panel = new wxPanel(this);
|
|
|
|
m_label_view_type = new wxStaticText(m_bottom_toolbar_panel, wxID_ANY, _L("View"));
|
|
|
|
m_choice_view_type = new wxChoice(m_bottom_toolbar_panel, wxID_ANY);
|
2020-04-29 11:51:20 +00:00
|
|
|
m_choice_view_type->Append(_L("Feature type"));
|
|
|
|
m_choice_view_type->Append(_L("Height"));
|
|
|
|
m_choice_view_type->Append(_L("Width"));
|
|
|
|
m_choice_view_type->Append(_L("Speed"));
|
|
|
|
m_choice_view_type->Append(_L("Fan speed"));
|
|
|
|
m_choice_view_type->Append(_L("Volumetric flow rate"));
|
|
|
|
m_choice_view_type->Append(_L("Tool"));
|
|
|
|
m_choice_view_type->Append(_L("Color Print"));
|
2018-09-18 11:35:05 +00:00
|
|
|
m_choice_view_type->SetSelection(0);
|
|
|
|
|
2020-05-19 08:04:14 +00:00
|
|
|
m_label_show = new wxStaticText(m_bottom_toolbar_panel, wxID_ANY, _L("Show"));
|
2018-09-18 11:35:05 +00:00
|
|
|
|
|
|
|
m_combochecklist_features = new wxComboCtrl();
|
2020-05-19 08:04:14 +00:00
|
|
|
m_combochecklist_features->Create(m_bottom_toolbar_panel, wxID_ANY, _L("Feature types"), wxDefaultPosition, wxDefaultSize, wxCB_READONLY);
|
2018-09-19 08:34:21 +00:00
|
|
|
std::string feature_items = GUI::into_u8(
|
2020-04-24 14:12:38 +00:00
|
|
|
_L("Unknown") + "|1|" +
|
2020-04-29 11:51:20 +00:00
|
|
|
_L("Perimeter") + "|1|" +
|
|
|
|
_L("External perimeter") + "|1|" +
|
|
|
|
_L("Overhang perimeter") + "|1|" +
|
|
|
|
_L("Internal infill") + "|1|" +
|
|
|
|
_L("Solid infill") + "|1|" +
|
|
|
|
_L("Top solid infill") + "|1|" +
|
|
|
|
_L("Ironing") + "|1|" +
|
|
|
|
_L("Bridge infill") + "|1|" +
|
|
|
|
_L("Gap fill") + "|1|" +
|
|
|
|
_L("Skirt") + "|1|" +
|
|
|
|
_L("Support material") + "|1|" +
|
|
|
|
_L("Support material interface") + "|1|" +
|
|
|
|
_L("Wipe tower") + "|1|" +
|
|
|
|
_L("Custom") + "|1"
|
2018-09-19 08:34:21 +00:00
|
|
|
);
|
2020-04-29 11:51:20 +00:00
|
|
|
Slic3r::GUI::create_combochecklist(m_combochecklist_features, GUI::into_u8(_L("Feature types")), feature_items);
|
2018-09-18 11:35:05 +00:00
|
|
|
|
2020-04-24 14:12:38 +00:00
|
|
|
m_combochecklist_options = new wxComboCtrl();
|
2020-05-19 08:04:14 +00:00
|
|
|
m_combochecklist_options->Create(m_bottom_toolbar_panel, wxID_ANY, _L("Options"), wxDefaultPosition, wxDefaultSize, wxCB_READONLY);
|
2020-04-24 14:12:38 +00:00
|
|
|
std::string options_items = GUI::into_u8(
|
2020-05-18 11:24:07 +00:00
|
|
|
get_option_type_string(OptionType::Travel) + "|0|" +
|
2020-11-21 09:36:10 +00:00
|
|
|
get_option_type_string(OptionType::Wipe) + "|0|" +
|
2020-05-18 11:24:07 +00:00
|
|
|
get_option_type_string(OptionType::Retractions) + "|0|" +
|
|
|
|
get_option_type_string(OptionType::Unretractions) + "|0|" +
|
|
|
|
get_option_type_string(OptionType::ToolChanges) + "|0|" +
|
|
|
|
get_option_type_string(OptionType::ColorChanges) + "|0|" +
|
|
|
|
get_option_type_string(OptionType::PausePrints) + "|0|" +
|
|
|
|
get_option_type_string(OptionType::CustomGCodes) + "|0|" +
|
|
|
|
get_option_type_string(OptionType::Shells) + "|0|" +
|
2020-11-03 08:27:18 +00:00
|
|
|
get_option_type_string(OptionType::ToolMarker) + "|1|" +
|
2020-08-05 13:43:46 +00:00
|
|
|
get_option_type_string(OptionType::Legend) + "|1"
|
2020-11-19 11:29:19 +00:00
|
|
|
);
|
2020-04-29 11:51:20 +00:00
|
|
|
Slic3r::GUI::create_combochecklist(m_combochecklist_options, GUI::into_u8(_L("Options")), options_items);
|
2020-11-19 11:29:19 +00:00
|
|
|
|
2020-08-19 10:59:50 +00:00
|
|
|
m_left_sizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
m_left_sizer->Add(m_canvas_widget, 1, wxALL | wxEXPAND, 0);
|
|
|
|
|
|
|
|
wxBoxSizer* right_sizer = new wxBoxSizer(wxVERTICAL);
|
|
|
|
right_sizer->Add(m_layers_slider_sizer, 1, wxEXPAND, 0);
|
2020-05-15 07:22:51 +00:00
|
|
|
|
2020-11-20 16:21:37 +00:00
|
|
|
m_moves_slider = new DoubleSlider::Control(m_bottom_toolbar_panel, wxID_ANY, 0, 0, 0, 100, wxDefaultPosition, wxDefaultSize, wxSL_HORIZONTAL);
|
2020-05-15 10:25:38 +00:00
|
|
|
m_moves_slider->SetDrawMode(DoubleSlider::dmSequentialGCodeView);
|
|
|
|
|
2020-05-19 08:04:14 +00:00
|
|
|
wxBoxSizer* bottom_toolbar_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
bottom_toolbar_sizer->AddSpacer(5);
|
|
|
|
bottom_toolbar_sizer->Add(m_label_view_type, 0, wxALIGN_CENTER_VERTICAL | wxRIGHT, 5);
|
|
|
|
bottom_toolbar_sizer->Add(m_choice_view_type, 0, wxALIGN_CENTER_VERTICAL, 0);
|
|
|
|
bottom_toolbar_sizer->AddSpacer(5);
|
|
|
|
bottom_toolbar_sizer->Add(m_label_show, 0, wxALIGN_CENTER_VERTICAL | wxLEFT | wxRIGHT, 5);
|
|
|
|
bottom_toolbar_sizer->Add(m_combochecklist_options, 0, wxALIGN_CENTER_VERTICAL, 0);
|
|
|
|
// change the following number if editing the layout of the bottom toolbar sizer. It is used into update_bottom_toolbar()
|
|
|
|
m_combochecklist_features_pos = 6;
|
|
|
|
bottom_toolbar_sizer->Add(m_combochecklist_features, 0, wxALIGN_CENTER_VERTICAL | wxLEFT, 5);
|
|
|
|
bottom_toolbar_sizer->Hide(m_combochecklist_features);
|
|
|
|
bottom_toolbar_sizer->AddSpacer(5);
|
|
|
|
bottom_toolbar_sizer->Add(m_moves_slider, 1, wxALL | wxEXPAND, 0);
|
|
|
|
m_bottom_toolbar_panel->SetSizer(bottom_toolbar_sizer);
|
2020-08-19 10:59:50 +00:00
|
|
|
|
|
|
|
m_left_sizer->Add(m_bottom_toolbar_panel, 0, wxALL | wxEXPAND, 0);
|
|
|
|
m_left_sizer->Hide(m_bottom_toolbar_panel);
|
|
|
|
|
|
|
|
wxBoxSizer* main_sizer = new wxBoxSizer(wxHORIZONTAL);
|
|
|
|
main_sizer->Add(m_left_sizer, 1, wxALL | wxEXPAND, 0);
|
|
|
|
main_sizer->Add(right_sizer, 0, wxALL | wxEXPAND, 0);
|
|
|
|
|
2018-09-18 11:35:05 +00:00
|
|
|
SetSizer(main_sizer);
|
|
|
|
SetMinSize(GetSize());
|
|
|
|
GetSizer()->SetSizeHints(this);
|
|
|
|
|
|
|
|
bind_event_handlers();
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Preview::~Preview()
|
|
|
|
{
|
|
|
|
unbind_event_handlers();
|
|
|
|
|
2020-03-02 09:58:46 +00:00
|
|
|
if (m_canvas != nullptr)
|
|
|
|
delete m_canvas;
|
|
|
|
|
2018-11-23 11:47:32 +00:00
|
|
|
if (m_canvas_widget != nullptr)
|
|
|
|
delete m_canvas_widget;
|
2018-09-18 11:35:05 +00:00
|
|
|
}
|
|
|
|
|
2019-06-24 13:55:14 +00:00
|
|
|
void Preview::set_as_dirty()
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->set_as_dirty();
|
|
|
|
}
|
|
|
|
|
2020-12-04 11:31:50 +00:00
|
|
|
#if !ENABLE_PREVIEW_TYPE_CHANGE
|
2018-09-18 11:35:05 +00:00
|
|
|
void Preview::set_number_extruders(unsigned int number_extruders)
|
|
|
|
{
|
2020-11-19 11:29:19 +00:00
|
|
|
if (m_number_extruders != number_extruders) {
|
2018-09-18 11:35:05 +00:00
|
|
|
m_number_extruders = number_extruders;
|
2018-12-18 08:08:32 +00:00
|
|
|
int tool_idx = m_choice_view_type->FindString(_(L("Tool")));
|
|
|
|
int type = (number_extruders > 1) ? tool_idx /* color by a tool number */ : 0; // color by a feature type
|
|
|
|
m_choice_view_type->SetSelection(type);
|
2020-11-19 11:29:19 +00:00
|
|
|
if (0 <= type && (type < static_cast<int>(GCodeViewer::EViewType::Count)))
|
2020-05-07 08:49:12 +00:00
|
|
|
m_canvas->set_gcode_view_preview_type(static_cast<GCodeViewer::EViewType>(type));
|
2018-12-18 08:08:32 +00:00
|
|
|
|
|
|
|
m_preferred_color_mode = (type == tool_idx) ? "tool_or_feature" : "feature";
|
2018-09-18 11:35:05 +00:00
|
|
|
}
|
|
|
|
}
|
2020-12-04 11:31:50 +00:00
|
|
|
#endif // !ENABLE_PREVIEW_TYPE_CHANGE
|
2018-09-18 11:35:05 +00:00
|
|
|
|
2019-02-19 14:15:27 +00:00
|
|
|
void Preview::bed_shape_changed()
|
|
|
|
{
|
|
|
|
if (m_canvas != nullptr)
|
|
|
|
m_canvas->bed_shape_changed();
|
|
|
|
}
|
2018-09-18 11:35:05 +00:00
|
|
|
|
|
|
|
void Preview::select_view(const std::string& direction)
|
|
|
|
{
|
2018-11-23 11:47:32 +00:00
|
|
|
m_canvas->select_view(direction);
|
2018-09-18 11:35:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Preview::set_drop_target(wxDropTarget* target)
|
|
|
|
{
|
|
|
|
if (target != nullptr)
|
|
|
|
SetDropTarget(target);
|
|
|
|
}
|
|
|
|
|
2019-03-19 09:48:20 +00:00
|
|
|
void Preview::load_print(bool keep_z_range)
|
2018-09-18 11:35:05 +00:00
|
|
|
{
|
2018-11-26 14:16:35 +00:00
|
|
|
PrinterTechnology tech = m_process->current_printer_technology();
|
|
|
|
if (tech == ptFFF)
|
2019-03-19 09:48:20 +00:00
|
|
|
load_print_as_fff(keep_z_range);
|
2018-11-26 14:16:35 +00:00
|
|
|
else if (tech == ptSLA)
|
|
|
|
load_print_as_sla();
|
2019-10-30 07:28:40 +00:00
|
|
|
|
2020-04-28 06:50:52 +00:00
|
|
|
update_bottom_toolbar();
|
2019-10-30 07:28:40 +00:00
|
|
|
Layout();
|
2018-09-18 11:35:05 +00:00
|
|
|
}
|
|
|
|
|
2019-03-18 13:07:55 +00:00
|
|
|
void Preview::reload_print(bool keep_volumes)
|
2018-09-18 11:35:05 +00:00
|
|
|
{
|
2019-03-21 10:44:39 +00:00
|
|
|
#ifdef __linux__
|
|
|
|
// We are getting mysterious crashes on Linux in gtk due to OpenGL context activation GH #1874 #1955.
|
|
|
|
// So we are applying a workaround here: a delayed release of OpenGL vertex buffers.
|
2019-03-18 13:07:55 +00:00
|
|
|
if (!IsShown())
|
|
|
|
{
|
|
|
|
m_volumes_cleanup_required = !keep_volumes;
|
2019-03-18 11:59:52 +00:00
|
|
|
return;
|
2019-03-18 13:07:55 +00:00
|
|
|
}
|
2021-01-11 12:31:30 +00:00
|
|
|
#endif /* __linux__ */
|
2019-03-21 10:54:25 +00:00
|
|
|
if (
|
2019-03-19 09:54:54 +00:00
|
|
|
#ifdef __linux__
|
2019-03-21 10:54:25 +00:00
|
|
|
m_volumes_cleanup_required ||
|
|
|
|
#endif /* __linux__ */
|
|
|
|
!keep_volumes)
|
2019-02-28 14:27:03 +00:00
|
|
|
{
|
|
|
|
m_canvas->reset_volumes();
|
|
|
|
m_loaded = false;
|
2019-03-21 10:54:25 +00:00
|
|
|
#ifdef __linux__
|
2019-03-18 13:07:55 +00:00
|
|
|
m_volumes_cleanup_required = false;
|
2019-03-21 10:54:25 +00:00
|
|
|
#endif /* __linux__ */
|
2019-02-28 14:27:03 +00:00
|
|
|
}
|
2018-09-18 11:35:05 +00:00
|
|
|
|
|
|
|
load_print();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Preview::refresh_print()
|
|
|
|
{
|
|
|
|
m_loaded = false;
|
|
|
|
|
|
|
|
if (!IsShown())
|
|
|
|
return;
|
|
|
|
|
2019-03-19 09:48:20 +00:00
|
|
|
load_print(true);
|
2018-09-18 11:35:05 +00:00
|
|
|
}
|
|
|
|
|
2019-04-24 23:45:00 +00:00
|
|
|
void Preview::msw_rescale()
|
2019-04-16 08:05:45 +00:00
|
|
|
{
|
2019-04-16 10:11:48 +00:00
|
|
|
// rescale slider
|
2020-05-15 10:25:38 +00:00
|
|
|
if (m_layers_slider != nullptr) m_layers_slider->msw_rescale();
|
|
|
|
if (m_moves_slider != nullptr) m_moves_slider->msw_rescale();
|
2019-04-16 10:11:48 +00:00
|
|
|
|
|
|
|
// rescale warning legend on the canvas
|
2019-04-24 23:45:00 +00:00
|
|
|
get_canvas3d()->msw_rescale();
|
2019-04-16 10:11:48 +00:00
|
|
|
|
|
|
|
// rescale legend
|
|
|
|
refresh_print();
|
2019-04-16 08:05:45 +00:00
|
|
|
}
|
|
|
|
|
2021-01-22 20:44:15 +00:00
|
|
|
void Preview::sys_color_changed()
|
|
|
|
{
|
|
|
|
if (m_layers_slider != nullptr)
|
|
|
|
m_layers_slider->sys_color_changed();
|
|
|
|
}
|
|
|
|
|
2020-11-04 15:26:25 +00:00
|
|
|
void Preview::jump_layers_slider(wxKeyEvent& evt)
|
|
|
|
{
|
|
|
|
if (m_layers_slider) m_layers_slider->OnChar(evt);
|
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::move_layers_slider(wxKeyEvent& evt)
|
|
|
|
{
|
|
|
|
if (m_layers_slider != nullptr) m_layers_slider->OnKeyDown(evt);
|
|
|
|
}
|
2019-06-13 14:10:33 +00:00
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::edit_layers_slider(wxKeyEvent& evt)
|
|
|
|
{
|
|
|
|
if (m_layers_slider != nullptr) m_layers_slider->OnChar(evt);
|
|
|
|
}
|
2019-06-18 09:40:26 +00:00
|
|
|
|
2018-09-18 11:35:05 +00:00
|
|
|
void Preview::bind_event_handlers()
|
|
|
|
{
|
|
|
|
this->Bind(wxEVT_SIZE, &Preview::on_size, this);
|
|
|
|
m_choice_view_type->Bind(wxEVT_CHOICE, &Preview::on_choice_view_type, this);
|
|
|
|
m_combochecklist_features->Bind(wxEVT_CHECKLISTBOX, &Preview::on_combochecklist_features, this);
|
2020-04-24 14:12:38 +00:00
|
|
|
m_combochecklist_options->Bind(wxEVT_CHECKLISTBOX, &Preview::on_combochecklist_options, this);
|
2020-05-19 08:04:14 +00:00
|
|
|
m_moves_slider->Bind(wxEVT_SCROLL_CHANGED, &Preview::on_moves_slider_scroll_changed, this);
|
2018-09-18 11:35:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Preview::unbind_event_handlers()
|
|
|
|
{
|
|
|
|
this->Unbind(wxEVT_SIZE, &Preview::on_size, this);
|
|
|
|
m_choice_view_type->Unbind(wxEVT_CHOICE, &Preview::on_choice_view_type, this);
|
|
|
|
m_combochecklist_features->Unbind(wxEVT_CHECKLISTBOX, &Preview::on_combochecklist_features, this);
|
2020-04-24 14:12:38 +00:00
|
|
|
m_combochecklist_options->Unbind(wxEVT_CHECKLISTBOX, &Preview::on_combochecklist_options, this);
|
2020-05-19 08:04:14 +00:00
|
|
|
m_moves_slider->Unbind(wxEVT_SCROLL_CHANGED, &Preview::on_moves_slider_scroll_changed, this);
|
2018-09-18 11:35:05 +00:00
|
|
|
}
|
|
|
|
|
2020-12-01 07:33:06 +00:00
|
|
|
#if ENABLE_ARROW_KEYS_WITH_SLIDERS
|
|
|
|
void Preview::move_moves_slider(wxKeyEvent& evt)
|
|
|
|
{
|
|
|
|
if (m_moves_slider != nullptr) m_moves_slider->OnKeyDown(evt);
|
|
|
|
}
|
|
|
|
#endif // ENABLE_ARROW_KEYS_WITH_SLIDERS
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::hide_layers_slider()
|
2020-05-15 07:22:51 +00:00
|
|
|
{
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider_sizer->Hide((size_t)0);
|
2020-05-15 07:22:51 +00:00
|
|
|
Layout();
|
|
|
|
}
|
2018-09-18 11:35:05 +00:00
|
|
|
|
|
|
|
void Preview::on_size(wxSizeEvent& evt)
|
|
|
|
{
|
|
|
|
evt.Skip();
|
|
|
|
Refresh();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Preview::on_choice_view_type(wxCommandEvent& evt)
|
|
|
|
{
|
2020-12-04 11:31:50 +00:00
|
|
|
#if !ENABLE_PREVIEW_TYPE_CHANGE
|
2018-09-18 11:35:05 +00:00
|
|
|
m_preferred_color_mode = (m_choice_view_type->GetStringSelection() == L("Tool")) ? "tool" : "feature";
|
2020-12-04 11:31:50 +00:00
|
|
|
#endif // !ENABLE_PREVIEW_TYPE_CHANGE
|
2018-09-18 11:35:05 +00:00
|
|
|
int selection = m_choice_view_type->GetCurrentSelection();
|
2020-12-04 11:31:50 +00:00
|
|
|
#if ENABLE_PREVIEW_TYPE_CHANGE
|
|
|
|
if (0 <= selection && selection < static_cast<int>(GCodeViewer::EViewType::Count)) {
|
|
|
|
m_canvas->set_toolpath_view_type(static_cast<GCodeViewer::EViewType>(selection));
|
|
|
|
m_keep_current_preview_type = true;
|
|
|
|
}
|
|
|
|
#else
|
2020-04-16 13:59:36 +00:00
|
|
|
if (0 <= selection && selection < static_cast<int>(GCodeViewer::EViewType::Count))
|
|
|
|
m_canvas->set_toolpath_view_type(static_cast<GCodeViewer::EViewType>(selection));
|
2020-12-04 11:31:50 +00:00
|
|
|
#endif // ENABLE_PREVIEW_TYPE_CHANGE
|
2020-04-16 13:59:36 +00:00
|
|
|
|
|
|
|
refresh_print();
|
2018-09-18 11:35:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Preview::on_combochecklist_features(wxCommandEvent& evt)
|
|
|
|
{
|
2020-04-28 06:50:52 +00:00
|
|
|
unsigned int flags = Slic3r::GUI::combochecklist_get_flags(m_combochecklist_features);
|
|
|
|
m_canvas->set_toolpath_role_visibility_flags(flags);
|
2018-09-18 11:35:05 +00:00
|
|
|
refresh_print();
|
|
|
|
}
|
|
|
|
|
2020-04-24 14:12:38 +00:00
|
|
|
void Preview::on_combochecklist_options(wxCommandEvent& evt)
|
|
|
|
{
|
2020-05-18 11:24:07 +00:00
|
|
|
unsigned int curr_flags = m_canvas->get_gcode_options_visibility_flags();
|
|
|
|
unsigned int new_flags = Slic3r::GUI::combochecklist_get_flags(m_combochecklist_options);
|
|
|
|
if (curr_flags == new_flags)
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_canvas->set_gcode_options_visibility_from_flags(new_flags);
|
|
|
|
|
2020-12-11 13:59:58 +00:00
|
|
|
#if ENABLE_RENDER_PATH_REFRESH_AFTER_OPTIONS_CHANGE
|
|
|
|
m_canvas->refresh_gcode_preview_render_paths();
|
|
|
|
#else
|
2021-01-29 06:39:27 +00:00
|
|
|
auto xored = [](unsigned int flags1, unsigned int flags2, unsigned int flag) {
|
|
|
|
auto is_flag_set = [](unsigned int flags, unsigned int flag) {
|
|
|
|
return (flags & (1 << flag)) != 0;
|
|
|
|
};
|
|
|
|
return !is_flag_set(flags1, flag) != !is_flag_set(flags2, flag);
|
|
|
|
};
|
|
|
|
|
2020-05-18 11:32:07 +00:00
|
|
|
bool skip_refresh = xored(curr_flags, new_flags, static_cast<unsigned int>(OptionType::Shells)) ||
|
|
|
|
xored(curr_flags, new_flags, static_cast<unsigned int>(OptionType::ToolMarker));
|
2020-05-18 11:24:07 +00:00
|
|
|
|
|
|
|
if (!skip_refresh)
|
|
|
|
refresh_print();
|
|
|
|
else
|
|
|
|
m_canvas->set_as_dirty();
|
2020-12-11 13:59:58 +00:00
|
|
|
#endif // ENABLE_RENDER_PATH_REFRESH_AFTER_OPTIONS_CHANGE
|
2020-04-24 14:12:38 +00:00
|
|
|
}
|
2019-06-22 15:38:14 +00:00
|
|
|
|
2020-12-04 11:31:50 +00:00
|
|
|
#if !ENABLE_PREVIEW_TYPE_CHANGE
|
2020-06-22 12:06:41 +00:00
|
|
|
void Preview::update_view_type(bool keep_volumes)
|
2019-07-02 11:13:17 +00:00
|
|
|
{
|
|
|
|
const DynamicPrintConfig& config = wxGetApp().preset_bundle->project_config;
|
|
|
|
|
2020-01-16 15:01:19 +00:00
|
|
|
const wxString& choice = !wxGetApp().plater()->model().custom_gcode_per_print_z.gcodes.empty() /*&&
|
2019-11-15 15:36:29 +00:00
|
|
|
(wxGetApp().extruders_edited_cnt()==1 || !slice_completed) */?
|
2020-06-09 14:12:57 +00:00
|
|
|
_L("Color Print") :
|
2019-10-10 14:03:58 +00:00
|
|
|
config.option<ConfigOptionFloats>("wiping_volumes_matrix")->values.size() > 1 ?
|
2020-06-09 14:12:57 +00:00
|
|
|
_L("Tool") :
|
|
|
|
_L("Feature type");
|
2019-07-02 11:13:17 +00:00
|
|
|
|
|
|
|
int type = m_choice_view_type->FindString(choice);
|
|
|
|
if (m_choice_view_type->GetSelection() != type) {
|
|
|
|
m_choice_view_type->SetSelection(type);
|
2020-11-19 11:29:19 +00:00
|
|
|
if (0 <= type && type < static_cast<int>(GCodeViewer::EViewType::Count))
|
2020-05-07 08:49:12 +00:00
|
|
|
m_canvas->set_gcode_view_preview_type(static_cast<GCodeViewer::EViewType>(type));
|
2019-07-02 11:13:17 +00:00
|
|
|
m_preferred_color_mode = "feature";
|
|
|
|
}
|
2020-06-09 14:12:57 +00:00
|
|
|
|
2020-06-22 12:06:41 +00:00
|
|
|
reload_print(keep_volumes);
|
2019-10-16 07:45:11 +00:00
|
|
|
}
|
2020-12-04 11:31:50 +00:00
|
|
|
#endif // !ENABLE_PREVIEW_TYPE_CHANGE
|
2019-10-16 07:45:11 +00:00
|
|
|
|
2020-04-28 06:50:52 +00:00
|
|
|
void Preview::update_bottom_toolbar()
|
|
|
|
{
|
|
|
|
combochecklist_set_flags(m_combochecklist_features, m_canvas->get_toolpath_role_visibility_flags());
|
|
|
|
combochecklist_set_flags(m_combochecklist_options, m_canvas->get_gcode_options_visibility_flags());
|
|
|
|
|
2020-05-19 08:04:14 +00:00
|
|
|
// updates visibility of features combobox
|
2020-11-19 11:29:19 +00:00
|
|
|
if (m_bottom_toolbar_panel->IsShown()) {
|
2020-05-19 08:04:14 +00:00
|
|
|
wxSizer* sizer = m_bottom_toolbar_panel->GetSizer();
|
|
|
|
bool show = !m_canvas->is_gcode_legend_enabled() || m_canvas->get_gcode_view_type() != GCodeViewer::EViewType::FeatureType;
|
|
|
|
|
2020-11-19 11:29:19 +00:00
|
|
|
if (show) {
|
|
|
|
if (sizer->GetItem(m_combochecklist_features) == nullptr) {
|
2020-05-19 08:04:14 +00:00
|
|
|
sizer->Insert(m_combochecklist_features_pos, m_combochecklist_features, 0, wxALIGN_CENTER_VERTICAL | wxLEFT, 5);
|
|
|
|
sizer->Show(m_combochecklist_features);
|
|
|
|
sizer->Layout();
|
|
|
|
Refresh();
|
|
|
|
}
|
|
|
|
}
|
2020-11-19 11:29:19 +00:00
|
|
|
else {
|
|
|
|
if (sizer->GetItem(m_combochecklist_features) != nullptr) {
|
2020-05-19 08:04:14 +00:00
|
|
|
sizer->Hide(m_combochecklist_features);
|
|
|
|
sizer->Detach(m_combochecklist_features);
|
|
|
|
sizer->Layout();
|
|
|
|
Refresh();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-04-28 06:50:52 +00:00
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
wxBoxSizer* Preview::create_layers_slider_sizer()
|
2020-05-15 07:22:51 +00:00
|
|
|
{
|
|
|
|
wxBoxSizer* sizer = new wxBoxSizer(wxHORIZONTAL);
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider = new DoubleSlider::Control(this, wxID_ANY, 0, 0, 0, 100);
|
2020-05-15 07:22:51 +00:00
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->SetDrawMode(wxGetApp().preset_bundle->printers.get_edited_preset().printer_technology() == ptSLA,
|
2020-05-15 07:22:51 +00:00
|
|
|
wxGetApp().preset_bundle->prints.get_edited_preset().config.opt_bool("complete_objects"));
|
2020-10-17 12:01:45 +00:00
|
|
|
m_layers_slider->enable_action_icon(wxGetApp().is_editor());
|
2020-05-15 07:22:51 +00:00
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
sizer->Add(m_layers_slider, 0, wxEXPAND, 0);
|
2020-05-15 07:22:51 +00:00
|
|
|
|
|
|
|
// sizer, m_canvas_widget
|
2020-05-15 10:25:38 +00:00
|
|
|
m_canvas_widget->Bind(wxEVT_KEY_DOWN, &Preview::update_layers_slider_from_canvas, this);
|
2020-05-15 07:22:51 +00:00
|
|
|
m_canvas_widget->Bind(wxEVT_KEY_UP, [this](wxKeyEvent& event) {
|
|
|
|
if (event.GetKeyCode() == WXK_SHIFT)
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->UseDefaultColors(true);
|
2020-05-15 07:22:51 +00:00
|
|
|
event.Skip();
|
|
|
|
});
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->Bind(wxEVT_SCROLL_CHANGED, &Preview::on_layers_slider_scroll_changed, this);
|
2020-05-15 07:22:51 +00:00
|
|
|
|
|
|
|
Bind(DoubleSlider::wxCUSTOMEVT_TICKSCHANGED, [this](wxEvent&) {
|
|
|
|
Model& model = wxGetApp().plater()->model();
|
2020-05-15 10:25:38 +00:00
|
|
|
model.custom_gcode_per_print_z = m_layers_slider->GetTicksValues();
|
2020-05-15 07:22:51 +00:00
|
|
|
m_schedule_background_process();
|
|
|
|
|
2020-12-04 11:31:50 +00:00
|
|
|
#if ENABLE_PREVIEW_TYPE_CHANGE
|
|
|
|
m_keep_current_preview_type = false;
|
|
|
|
reload_print(false);
|
|
|
|
#else
|
2020-05-15 07:22:51 +00:00
|
|
|
update_view_type(false);
|
2020-12-04 11:31:50 +00:00
|
|
|
#endif // ENABLE_PREVIEW_TYPE_CHANGE
|
2020-05-15 07:22:51 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
return sizer;
|
|
|
|
}
|
2018-10-05 21:29:15 +00:00
|
|
|
|
2018-12-20 09:55:50 +00:00
|
|
|
// Find an index of a value in a sorted vector, which is in <z-eps, z+eps>.
|
|
|
|
// Returns -1 if there is no such member.
|
|
|
|
static int find_close_layer_idx(const std::vector<double>& zs, double &z, double eps)
|
|
|
|
{
|
|
|
|
if (zs.empty())
|
|
|
|
return -1;
|
|
|
|
auto it_h = std::lower_bound(zs.begin(), zs.end(), z);
|
|
|
|
if (it_h == zs.end()) {
|
|
|
|
auto it_l = it_h;
|
|
|
|
-- it_l;
|
|
|
|
if (z - *it_l < eps)
|
|
|
|
return int(zs.size() - 1);
|
|
|
|
} else if (it_h == zs.begin()) {
|
|
|
|
if (*it_h - z < eps)
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
auto it_l = it_h;
|
|
|
|
-- it_l;
|
|
|
|
double dist_l = z - *it_l;
|
|
|
|
double dist_h = *it_h - z;
|
|
|
|
if (std::min(dist_l, dist_h) < eps) {
|
|
|
|
return (dist_l < dist_h) ? int(it_l - zs.begin()) : int(it_h - zs.begin());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::check_layers_slider_values(std::vector<CustomGCode::Item>& ticks_from_model, const std::vector<double>& layers_z)
|
2019-10-10 14:03:58 +00:00
|
|
|
{
|
|
|
|
// All ticks that would end up outside the slider range should be erased.
|
|
|
|
// TODO: this should be placed into more appropriate part of code,
|
|
|
|
// this function is e.g. not called when the last object is deleted
|
|
|
|
unsigned int old_size = ticks_from_model.size();
|
|
|
|
ticks_from_model.erase(std::remove_if(ticks_from_model.begin(), ticks_from_model.end(),
|
2020-01-23 11:49:39 +00:00
|
|
|
[layers_z](CustomGCode::Item val)
|
2019-10-10 14:03:58 +00:00
|
|
|
{
|
2019-12-17 07:37:50 +00:00
|
|
|
auto it = std::lower_bound(layers_z.begin(), layers_z.end(), val.print_z - DoubleSlider::epsilon());
|
2019-10-10 14:03:58 +00:00
|
|
|
return it == layers_z.end();
|
|
|
|
}),
|
|
|
|
ticks_from_model.end());
|
|
|
|
if (ticks_from_model.size() != old_size)
|
|
|
|
m_schedule_background_process();
|
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::update_layers_slider(const std::vector<double>& layers_z, bool keep_z_range)
|
2020-05-15 07:22:51 +00:00
|
|
|
{
|
|
|
|
// Save the initial slider span.
|
2020-05-15 10:25:38 +00:00
|
|
|
double z_low = m_layers_slider->GetLowerValueD();
|
|
|
|
double z_high = m_layers_slider->GetHigherValueD();
|
|
|
|
bool was_empty = m_layers_slider->GetMaxValue() == 0;
|
2020-05-15 07:22:51 +00:00
|
|
|
|
|
|
|
bool force_sliders_full_range = was_empty;
|
|
|
|
if (!keep_z_range)
|
|
|
|
{
|
2020-05-15 10:25:38 +00:00
|
|
|
bool span_changed = layers_z.empty() || std::abs(layers_z.back() - m_layers_slider->GetMaxValueD()) > DoubleSlider::epsilon()/*1e-6*/;
|
2020-05-15 07:22:51 +00:00
|
|
|
force_sliders_full_range |= span_changed;
|
2020-05-15 10:25:38 +00:00
|
|
|
}
|
|
|
|
bool snap_to_min = force_sliders_full_range || m_layers_slider->is_lower_at_min();
|
|
|
|
bool snap_to_max = force_sliders_full_range || m_layers_slider->is_higher_at_max();
|
2020-05-15 07:22:51 +00:00
|
|
|
|
|
|
|
// Detect and set manipulation mode for double slider
|
2020-05-15 10:25:38 +00:00
|
|
|
update_layers_slider_mode();
|
2020-05-15 07:22:51 +00:00
|
|
|
|
2020-11-19 09:57:24 +00:00
|
|
|
Plater* plater = wxGetApp().plater();
|
|
|
|
CustomGCode::Info& ticks_info_from_model = plater->model().custom_gcode_per_print_z;
|
2020-05-15 10:25:38 +00:00
|
|
|
check_layers_slider_values(ticks_info_from_model.gcodes, layers_z);
|
2020-05-15 07:22:51 +00:00
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->SetSliderValues(layers_z);
|
|
|
|
assert(m_layers_slider->GetMinValue() == 0);
|
|
|
|
m_layers_slider->SetMaxValue(layers_z.empty() ? 0 : layers_z.size() - 1);
|
2020-05-15 07:22:51 +00:00
|
|
|
|
|
|
|
int idx_low = 0;
|
2020-05-15 10:25:38 +00:00
|
|
|
int idx_high = m_layers_slider->GetMaxValue();
|
2020-05-15 07:22:51 +00:00
|
|
|
if (!layers_z.empty()) {
|
|
|
|
if (!snap_to_min) {
|
|
|
|
int idx_new = find_close_layer_idx(layers_z, z_low, DoubleSlider::epsilon()/*1e-6*/);
|
|
|
|
if (idx_new != -1)
|
|
|
|
idx_low = idx_new;
|
|
|
|
}
|
|
|
|
if (!snap_to_max) {
|
|
|
|
int idx_new = find_close_layer_idx(layers_z, z_high, DoubleSlider::epsilon()/*1e-6*/);
|
|
|
|
if (idx_new != -1)
|
|
|
|
idx_high = idx_new;
|
|
|
|
}
|
|
|
|
}
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->SetSelectionSpan(idx_low, idx_high);
|
|
|
|
m_layers_slider->SetTicksValues(ticks_info_from_model);
|
2020-05-15 07:22:51 +00:00
|
|
|
|
2020-11-19 09:57:24 +00:00
|
|
|
bool sla_print_technology = plater->printer_technology() == ptSLA;
|
2020-05-15 07:22:51 +00:00
|
|
|
bool sequential_print = wxGetApp().preset_bundle->prints.get_edited_preset().config.opt_bool("complete_objects");
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->SetDrawMode(sla_print_technology, sequential_print);
|
2020-11-19 09:57:24 +00:00
|
|
|
m_layers_slider->SetExtruderColors(plater->get_extruder_colors_from_plater_config());
|
|
|
|
if (sla_print_technology)
|
|
|
|
m_layers_slider->SetLayersTimes(plater->sla_print().print_statistics().layers_times);
|
|
|
|
else
|
2020-11-21 12:16:03 +00:00
|
|
|
m_layers_slider->SetLayersTimes(m_gcode_result->time_statistics.modes.front().layers_times);
|
2020-05-15 07:22:51 +00:00
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider_sizer->Show((size_t)0);
|
2020-05-15 07:22:51 +00:00
|
|
|
Layout();
|
|
|
|
}
|
2020-01-10 15:49:07 +00:00
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::update_layers_slider_mode()
|
2020-01-10 15:49:07 +00:00
|
|
|
{
|
|
|
|
// true -> single-extruder printer profile OR
|
|
|
|
// multi-extruder printer profile , but whole model is printed by only one extruder
|
|
|
|
// false -> multi-extruder printer profile , and model is printed by several extruders
|
|
|
|
bool one_extruder_printed_model = true;
|
|
|
|
|
|
|
|
// extruder used for whole model for multi-extruder printer profile
|
|
|
|
int only_extruder = -1;
|
|
|
|
|
|
|
|
if (wxGetApp().extruders_edited_cnt() > 1)
|
|
|
|
{
|
|
|
|
const ModelObjectPtrs& objects = wxGetApp().plater()->model().objects;
|
|
|
|
|
|
|
|
// check if whole model uses just only one extruder
|
|
|
|
if (!objects.empty())
|
|
|
|
{
|
|
|
|
const int extruder = objects[0]->config.has("extruder") ?
|
|
|
|
objects[0]->config.option("extruder")->getInt() : 0;
|
|
|
|
|
|
|
|
auto is_one_extruder_printed_model = [objects, extruder]()
|
|
|
|
{
|
|
|
|
for (ModelObject* object : objects)
|
|
|
|
{
|
|
|
|
if (object->config.has("extruder") &&
|
|
|
|
object->config.option("extruder")->getInt() != extruder)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (object->volumes.size() > 1)
|
|
|
|
for (ModelVolume* volume : object->volumes)
|
|
|
|
if (volume->config.has("extruder") &&
|
|
|
|
volume->config.option("extruder")->getInt() != extruder)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (const auto& range : object->layer_config_ranges)
|
|
|
|
if (range.second.has("extruder") &&
|
|
|
|
range.second.option("extruder")->getInt() != extruder)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
if (is_one_extruder_printed_model())
|
|
|
|
only_extruder = extruder;
|
|
|
|
else
|
|
|
|
one_extruder_printed_model = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->SetModeAndOnlyExtruder(one_extruder_printed_model, only_extruder);
|
2018-10-05 21:29:15 +00:00
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::reset_layers_slider()
|
2020-05-15 07:22:51 +00:00
|
|
|
{
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->SetHigherValue(0);
|
|
|
|
m_layers_slider->SetLowerValue(0);
|
2020-05-15 07:22:51 +00:00
|
|
|
}
|
2018-10-05 21:29:15 +00:00
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::update_layers_slider_from_canvas(wxKeyEvent& event)
|
2020-05-15 07:22:51 +00:00
|
|
|
{
|
|
|
|
if (event.HasModifiers()) {
|
|
|
|
event.Skip();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto key = event.GetKeyCode();
|
|
|
|
|
2020-12-01 07:33:06 +00:00
|
|
|
#if ENABLE_ARROW_KEYS_WITH_SLIDERS
|
|
|
|
if (key == 'S' || key == 'W') {
|
|
|
|
const int new_pos = key == 'W' ? m_layers_slider->GetHigherValue() + 1 : m_layers_slider->GetHigherValue() - 1;
|
|
|
|
#else
|
2020-05-15 07:22:51 +00:00
|
|
|
if (key == 'U' || key == 'D') {
|
2020-05-15 10:25:38 +00:00
|
|
|
const int new_pos = key == 'U' ? m_layers_slider->GetHigherValue() + 1 : m_layers_slider->GetHigherValue() - 1;
|
2020-12-01 07:33:06 +00:00
|
|
|
#endif // ENABLE_ARROW_KEYS_WITH_SLIDERS
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->SetHigherValue(new_pos);
|
|
|
|
if (event.ShiftDown() || m_layers_slider->is_one_layer()) m_layers_slider->SetLowerValue(m_layers_slider->GetHigherValue());
|
2020-05-15 07:22:51 +00:00
|
|
|
}
|
2020-12-01 07:33:06 +00:00
|
|
|
#if ENABLE_ARROW_KEYS_WITH_SLIDERS
|
|
|
|
else if (key == 'A' || key == 'D') {
|
|
|
|
const int new_pos = key == 'D' ? m_moves_slider->GetHigherValue() + 1 : m_moves_slider->GetHigherValue() - 1;
|
|
|
|
m_moves_slider->SetHigherValue(new_pos);
|
|
|
|
if (event.ShiftDown() || m_moves_slider->is_one_layer()) m_moves_slider->SetLowerValue(m_moves_slider->GetHigherValue());
|
|
|
|
}
|
|
|
|
else if (key == 'X')
|
|
|
|
#else
|
2020-05-15 07:22:51 +00:00
|
|
|
else if (key == 'S')
|
2020-12-01 07:33:06 +00:00
|
|
|
#endif // ENABLE_ARROW_KEYS_WITH_SLIDERS
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->ChangeOneLayerLock();
|
2020-05-15 07:22:51 +00:00
|
|
|
else if (key == WXK_SHIFT)
|
2020-05-15 10:25:38 +00:00
|
|
|
m_layers_slider->UseDefaultColors(false);
|
2020-05-15 07:22:51 +00:00
|
|
|
else
|
|
|
|
event.Skip();
|
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::update_moves_slider()
|
2020-05-15 07:22:51 +00:00
|
|
|
{
|
|
|
|
const GCodeViewer::SequentialView& view = m_canvas->get_gcode_sequential_view();
|
2020-05-28 09:14:56 +00:00
|
|
|
// this should not be needed, but it is here to try to prevent rambling crashes on Mac Asan
|
|
|
|
if (view.endpoints.last < view.endpoints.first)
|
|
|
|
return;
|
|
|
|
|
2020-05-18 11:24:07 +00:00
|
|
|
std::vector<double> values(view.endpoints.last - view.endpoints.first + 1);
|
2021-02-16 09:07:05 +00:00
|
|
|
#if ENABLE_GCODE_LINES_ID_IN_H_SLIDER
|
|
|
|
std::vector<double> alternate_values(view.endpoints.last - view.endpoints.first + 1);
|
|
|
|
#endif // ENABLE_GCODE_LINES_ID_IN_H_SLIDER
|
2020-05-15 07:22:51 +00:00
|
|
|
unsigned int count = 0;
|
2020-10-07 11:38:21 +00:00
|
|
|
for (unsigned int i = view.endpoints.first; i <= view.endpoints.last; ++i) {
|
2021-02-16 09:07:05 +00:00
|
|
|
#if ENABLE_GCODE_LINES_ID_IN_H_SLIDER
|
|
|
|
values[count] = static_cast<double>(i + 1);
|
|
|
|
if (view.gcode_ids[i] > 0)
|
|
|
|
alternate_values[count] = static_cast<double>(view.gcode_ids[i]);
|
|
|
|
++count;
|
|
|
|
#else
|
2020-05-15 07:22:51 +00:00
|
|
|
values[count++] = static_cast<double>(i + 1);
|
2021-02-16 09:07:05 +00:00
|
|
|
#endif // ENABLE_GCODE_LINES_ID_IN_H_SLIDER
|
2020-05-15 07:22:51 +00:00
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
m_moves_slider->SetSliderValues(values);
|
2021-02-16 09:07:05 +00:00
|
|
|
#if ENABLE_GCODE_LINES_ID_IN_H_SLIDER
|
|
|
|
m_moves_slider->SetSliderAlternateValues(alternate_values);
|
|
|
|
#endif // ENABLE_GCODE_LINES_ID_IN_H_SLIDER
|
2020-05-18 11:24:07 +00:00
|
|
|
m_moves_slider->SetMaxValue(view.endpoints.last - view.endpoints.first);
|
|
|
|
m_moves_slider->SetSelectionSpan(view.current.first - view.endpoints.first, view.current.last - view.endpoints.first);
|
2020-05-15 07:22:51 +00:00
|
|
|
}
|
2020-10-07 11:38:21 +00:00
|
|
|
|
|
|
|
void Preview::enable_moves_slider(bool enable)
|
|
|
|
{
|
2020-10-08 12:09:39 +00:00
|
|
|
bool render_as_disabled = !enable;
|
|
|
|
if (m_moves_slider != nullptr && m_moves_slider->is_rendering_as_disabled() != render_as_disabled) {
|
|
|
|
m_moves_slider->set_render_as_disabled(render_as_disabled);
|
2020-10-08 10:50:29 +00:00
|
|
|
m_moves_slider->Refresh();
|
|
|
|
}
|
2020-10-07 11:38:21 +00:00
|
|
|
}
|
2018-10-05 21:29:15 +00:00
|
|
|
|
2019-03-19 09:48:20 +00:00
|
|
|
void Preview::load_print_as_fff(bool keep_z_range)
|
2018-11-26 14:16:35 +00:00
|
|
|
{
|
2020-10-12 06:39:40 +00:00
|
|
|
if (wxGetApp().mainframe == nullptr || wxGetApp().is_recreating_gui())
|
|
|
|
// avoid processing while mainframe is being constructed
|
2020-06-05 14:50:17 +00:00
|
|
|
return;
|
|
|
|
|
2018-11-26 14:16:35 +00:00
|
|
|
if (m_loaded || m_process->current_printer_technology() != ptFFF)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// we require that there's at least one object and the posSlice step
|
|
|
|
// is performed on all of them(this ensures that _shifted_copies was
|
|
|
|
// populated and we know the number of layers)
|
2019-01-09 13:15:18 +00:00
|
|
|
bool has_layers = false;
|
2018-11-26 14:16:35 +00:00
|
|
|
const Print *print = m_process->fff_print();
|
2019-01-09 13:15:18 +00:00
|
|
|
if (print->is_step_done(posSlice)) {
|
2018-11-26 14:16:35 +00:00
|
|
|
for (const PrintObject* print_object : print->objects())
|
2019-01-09 13:15:18 +00:00
|
|
|
if (! print_object->layers().empty()) {
|
|
|
|
has_layers = true;
|
|
|
|
break;
|
2018-11-26 14:16:35 +00:00
|
|
|
}
|
2019-01-09 13:15:18 +00:00
|
|
|
}
|
|
|
|
if (print->is_step_done(posSupportMaterial)) {
|
|
|
|
for (const PrintObject* print_object : print->objects())
|
|
|
|
if (! print_object->support_layers().empty()) {
|
|
|
|
has_layers = true;
|
|
|
|
break;
|
2018-11-26 14:16:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-22 10:55:13 +00:00
|
|
|
if (wxGetApp().is_editor() && !has_layers) {
|
2020-05-15 10:25:38 +00:00
|
|
|
hide_layers_slider();
|
2020-08-19 10:59:50 +00:00
|
|
|
m_left_sizer->Hide(m_bottom_toolbar_panel);
|
|
|
|
m_left_sizer->Layout();
|
2020-06-05 14:50:17 +00:00
|
|
|
Refresh();
|
2018-11-26 14:16:35 +00:00
|
|
|
m_canvas_widget->Refresh();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-12-04 11:31:50 +00:00
|
|
|
#if !ENABLE_PREVIEW_TYPE_CHANGE
|
2020-10-22 10:55:13 +00:00
|
|
|
if (m_preferred_color_mode == "tool_or_feature") {
|
2018-11-26 14:16:35 +00:00
|
|
|
// It is left to Slic3r to decide whether the print shall be colored by the tool or by the feature.
|
|
|
|
// Color by feature if it is a single extruder print.
|
|
|
|
unsigned int number_extruders = (unsigned int)print->extruders().size();
|
2020-11-02 10:59:37 +00:00
|
|
|
int tool_idx = m_choice_view_type->FindString(_L("Tool"));
|
2018-11-26 14:16:35 +00:00
|
|
|
int type = (number_extruders > 1) ? tool_idx /* color by a tool number */ : 0; // color by a feature type
|
|
|
|
m_choice_view_type->SetSelection(type);
|
2020-04-22 14:29:07 +00:00
|
|
|
if (0 <= type && type < static_cast<int>(GCodeViewer::EViewType::Count))
|
|
|
|
m_canvas->set_gcode_view_preview_type(static_cast<GCodeViewer::EViewType>(type));
|
2018-11-26 14:16:35 +00:00
|
|
|
// If the->SetSelection changed the following line, revert it to "decide yourself".
|
|
|
|
m_preferred_color_mode = "tool_or_feature";
|
|
|
|
}
|
2020-12-04 11:31:50 +00:00
|
|
|
#endif // !ENABLE_PREVIEW_TYPE_CHANGE
|
2018-11-26 14:16:35 +00:00
|
|
|
|
2020-04-22 14:29:07 +00:00
|
|
|
GCodeViewer::EViewType gcode_view_type = m_canvas->get_gcode_view_preview_type();
|
2020-06-05 14:50:17 +00:00
|
|
|
bool gcode_preview_data_valid = !m_gcode_result->moves.empty();
|
2018-11-26 14:16:35 +00:00
|
|
|
// Collect colors per extruder.
|
|
|
|
std::vector<std::string> colors;
|
2020-01-23 11:49:39 +00:00
|
|
|
std::vector<CustomGCode::Item> color_print_values = {};
|
2019-01-29 14:11:29 +00:00
|
|
|
// set color print values, if it si selected "ColorPrint" view type
|
2020-10-22 10:55:13 +00:00
|
|
|
if (gcode_view_type == GCodeViewer::EViewType::ColorPrint) {
|
2020-11-09 12:06:20 +00:00
|
|
|
colors = wxGetApp().plater()->get_colors_for_color_print(m_gcode_result);
|
2019-11-04 12:42:47 +00:00
|
|
|
|
2020-10-22 10:55:13 +00:00
|
|
|
if (!gcode_preview_data_valid) {
|
2020-01-16 15:01:19 +00:00
|
|
|
color_print_values = wxGetApp().plater()->model().custom_gcode_per_print_z.gcodes;
|
2020-10-22 10:55:13 +00:00
|
|
|
colors.push_back("#808080"); // gray color for pause print or custom G-code
|
|
|
|
}
|
2019-01-29 14:11:29 +00:00
|
|
|
}
|
2020-10-22 10:55:13 +00:00
|
|
|
else if (gcode_preview_data_valid || gcode_view_type == GCodeViewer::EViewType::Tool) {
|
2020-11-09 12:06:20 +00:00
|
|
|
colors = wxGetApp().plater()->get_extruder_colors_from_plater_config(m_gcode_result);
|
2019-01-30 11:10:26 +00:00
|
|
|
color_print_values.clear();
|
2018-11-26 14:16:35 +00:00
|
|
|
}
|
|
|
|
|
2020-10-22 10:55:13 +00:00
|
|
|
if (IsShown()) {
|
2020-04-29 08:18:29 +00:00
|
|
|
std::vector<double> zs;
|
|
|
|
|
2019-11-27 15:03:16 +00:00
|
|
|
m_canvas->set_selected_extruder(0);
|
2019-03-14 15:31:55 +00:00
|
|
|
if (gcode_preview_data_valid) {
|
2019-01-15 09:00:34 +00:00
|
|
|
// Load the real G-code preview.
|
2020-04-22 14:29:07 +00:00
|
|
|
m_canvas->load_gcode_preview(*m_gcode_result);
|
|
|
|
m_canvas->refresh_gcode_preview(*m_gcode_result, colors);
|
2020-08-19 10:59:50 +00:00
|
|
|
m_left_sizer->Show(m_bottom_toolbar_panel);
|
|
|
|
m_left_sizer->Layout();
|
2020-05-19 08:04:14 +00:00
|
|
|
Refresh();
|
2020-04-29 08:18:29 +00:00
|
|
|
zs = m_canvas->get_gcode_layers_zs();
|
2019-03-14 15:31:55 +00:00
|
|
|
m_loaded = true;
|
2019-07-02 11:13:17 +00:00
|
|
|
} else {
|
2019-01-15 09:00:34 +00:00
|
|
|
// Load the initial preview based on slices, not the final G-code.
|
2019-01-29 14:11:29 +00:00
|
|
|
m_canvas->load_preview(colors, color_print_values);
|
2020-08-19 10:59:50 +00:00
|
|
|
m_left_sizer->Hide(m_bottom_toolbar_panel);
|
|
|
|
m_left_sizer->Layout();
|
2020-05-19 08:04:14 +00:00
|
|
|
Refresh();
|
2020-04-29 08:18:29 +00:00
|
|
|
zs = m_canvas->get_volumes_print_zs(true);
|
2019-07-02 11:13:17 +00:00
|
|
|
}
|
2019-01-15 09:00:34 +00:00
|
|
|
if (zs.empty()) {
|
|
|
|
// all layers filtered out
|
2020-05-15 10:25:38 +00:00
|
|
|
hide_layers_slider();
|
2019-01-15 09:00:34 +00:00
|
|
|
m_canvas_widget->Refresh();
|
|
|
|
} else
|
2020-05-15 10:25:38 +00:00
|
|
|
update_layers_slider(zs, keep_z_range);
|
2018-11-26 14:16:35 +00:00
|
|
|
}
|
2020-12-04 11:31:50 +00:00
|
|
|
|
|
|
|
#if ENABLE_PREVIEW_TYPE_CHANGE
|
|
|
|
unsigned int number_extruders = (unsigned int)print->extruders().size();
|
|
|
|
|
|
|
|
if (!m_keep_current_preview_type) {
|
|
|
|
const wxString choice = !wxGetApp().plater()->model().custom_gcode_per_print_z.gcodes.empty() ?
|
|
|
|
_L("Color Print") :
|
|
|
|
(number_extruders > 1) ? _L("Tool") : _L("Feature type");
|
|
|
|
|
|
|
|
int type = m_choice_view_type->FindString(choice);
|
|
|
|
if (m_choice_view_type->GetSelection() != type) {
|
|
|
|
if (0 <= type && type < static_cast<int>(GCodeViewer::EViewType::Count)) {
|
|
|
|
m_choice_view_type->SetSelection(type);
|
|
|
|
m_canvas->set_gcode_view_preview_type(static_cast<GCodeViewer::EViewType>(type));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif // ENABLE_PREVIEW_TYPE_CHANGE
|
2018-11-26 14:16:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Preview::load_print_as_sla()
|
|
|
|
{
|
|
|
|
if (m_loaded || (m_process->current_printer_technology() != ptSLA))
|
|
|
|
return;
|
|
|
|
|
2018-11-26 15:17:59 +00:00
|
|
|
unsigned int n_layers = 0;
|
|
|
|
const SLAPrint* print = m_process->sla_print();
|
|
|
|
|
2019-03-22 20:26:58 +00:00
|
|
|
std::vector<double> zs;
|
|
|
|
double initial_layer_height = print->material_config().initial_layer_height.value;
|
2018-11-26 15:17:59 +00:00
|
|
|
for (const SLAPrintObject* obj : print->objects())
|
2020-11-19 11:29:19 +00:00
|
|
|
if (obj->is_step_done(slaposSliceSupports) && !obj->get_slice_index().empty()) {
|
2019-03-25 18:02:05 +00:00
|
|
|
auto low_coord = obj->get_slice_index().front().print_level();
|
|
|
|
for (auto& rec : obj->get_slice_index())
|
|
|
|
zs.emplace_back(initial_layer_height + (rec.print_level() - low_coord) * SCALING_FACTOR);
|
2018-11-26 15:17:59 +00:00
|
|
|
}
|
2019-03-22 20:26:58 +00:00
|
|
|
sort_remove_duplicates(zs);
|
2018-11-26 15:17:59 +00:00
|
|
|
|
2019-04-02 11:47:49 +00:00
|
|
|
m_canvas->reset_clipping_planes_cache();
|
2021-01-28 07:28:23 +00:00
|
|
|
m_canvas->set_use_clipping_planes(true);
|
2019-04-02 11:47:49 +00:00
|
|
|
|
2018-11-26 15:17:59 +00:00
|
|
|
n_layers = (unsigned int)zs.size();
|
2020-11-19 11:29:19 +00:00
|
|
|
if (n_layers == 0) {
|
2020-05-15 10:25:38 +00:00
|
|
|
hide_layers_slider();
|
2018-11-26 15:17:59 +00:00
|
|
|
m_canvas_widget->Refresh();
|
|
|
|
}
|
|
|
|
|
2020-11-19 11:29:19 +00:00
|
|
|
if (IsShown()) {
|
2018-11-26 15:17:59 +00:00
|
|
|
m_canvas->load_sla_preview();
|
2020-08-19 10:59:50 +00:00
|
|
|
m_left_sizer->Hide(m_bottom_toolbar_panel);
|
|
|
|
m_left_sizer->Hide(m_bottom_toolbar_panel);
|
|
|
|
m_left_sizer->Layout();
|
2020-05-19 08:04:14 +00:00
|
|
|
Refresh();
|
2018-11-26 15:17:59 +00:00
|
|
|
|
|
|
|
if (n_layers > 0)
|
2020-05-15 10:25:38 +00:00
|
|
|
update_layers_slider(zs);
|
2018-11-26 15:17:59 +00:00
|
|
|
|
|
|
|
m_loaded = true;
|
|
|
|
}
|
2018-11-26 14:16:35 +00:00
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::on_layers_slider_scroll_changed(wxCommandEvent& event)
|
2018-11-27 13:50:57 +00:00
|
|
|
{
|
2020-11-11 15:22:09 +00:00
|
|
|
if (IsShown()) {
|
2018-11-27 13:50:57 +00:00
|
|
|
PrinterTechnology tech = m_process->current_printer_technology();
|
2020-11-11 15:22:09 +00:00
|
|
|
if (tech == ptFFF) {
|
2020-11-13 10:50:52 +00:00
|
|
|
m_canvas->set_volumes_z_range({ m_layers_slider->GetLowerValueD(), m_layers_slider->GetHigherValueD() });
|
|
|
|
m_canvas->set_toolpaths_z_range({ static_cast<unsigned int>(m_layers_slider->GetLowerValue()), static_cast<unsigned int>(m_layers_slider->GetHigherValue()) });
|
2020-04-27 09:44:29 +00:00
|
|
|
m_canvas->set_as_dirty();
|
2018-11-27 13:50:57 +00:00
|
|
|
}
|
2020-11-11 15:22:09 +00:00
|
|
|
else if (tech == ptSLA) {
|
2020-05-15 10:25:38 +00:00
|
|
|
m_canvas->set_clipping_plane(0, ClippingPlane(Vec3d::UnitZ(), -m_layers_slider->GetLowerValueD()));
|
|
|
|
m_canvas->set_clipping_plane(1, ClippingPlane(-Vec3d::UnitZ(), m_layers_slider->GetHigherValueD()));
|
2019-05-17 13:04:06 +00:00
|
|
|
m_canvas->render();
|
2018-11-27 13:50:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-15 10:25:38 +00:00
|
|
|
void Preview::on_moves_slider_scroll_changed(wxCommandEvent& event)
|
2020-05-15 07:22:51 +00:00
|
|
|
{
|
2020-06-23 12:31:08 +00:00
|
|
|
m_canvas->update_gcode_sequential_view_current(static_cast<unsigned int>(m_moves_slider->GetLowerValueD() - 1.0), static_cast<unsigned int>(m_moves_slider->GetHigherValueD() - 1.0));
|
2020-05-15 07:22:51 +00:00
|
|
|
m_canvas->render();
|
|
|
|
}
|
2020-05-18 11:24:07 +00:00
|
|
|
|
|
|
|
wxString Preview::get_option_type_string(OptionType type) const
|
|
|
|
{
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case OptionType::Travel: { return _L("Travel"); }
|
2020-11-21 09:36:10 +00:00
|
|
|
case OptionType::Wipe: { return _L("Wipe"); }
|
2020-05-18 11:24:07 +00:00
|
|
|
case OptionType::Retractions: { return _L("Retractions"); }
|
2020-10-17 19:23:24 +00:00
|
|
|
case OptionType::Unretractions: { return _L("Deretractions"); }
|
2020-05-18 11:24:07 +00:00
|
|
|
case OptionType::ToolChanges: { return _L("Tool changes"); }
|
|
|
|
case OptionType::ColorChanges: { return _L("Color changes"); }
|
2020-11-05 21:17:49 +00:00
|
|
|
case OptionType::PausePrints: { return _L("Print pauses"); }
|
|
|
|
case OptionType::CustomGCodes: { return _L("Custom G-codes"); }
|
2020-05-18 11:24:07 +00:00
|
|
|
case OptionType::Shells: { return _L("Shells"); }
|
|
|
|
case OptionType::ToolMarker: { return _L("Tool marker"); }
|
2020-08-06 08:15:34 +00:00
|
|
|
case OptionType::Legend: { return _L("Legend/Estimated printing time"); }
|
2020-05-18 11:24:07 +00:00
|
|
|
default: { return ""; }
|
|
|
|
}
|
|
|
|
}
|
2020-05-15 07:22:51 +00:00
|
|
|
|
2018-09-18 11:35:05 +00:00
|
|
|
} // namespace GUI
|
|
|
|
} // namespace Slic3r
|