2019-05-28 14:38:04 +00:00
|
|
|
#include "GUI_ObjectLayers.hpp"
|
|
|
|
#include "GUI_ObjectList.hpp"
|
|
|
|
|
|
|
|
#include "OptionsGroup.hpp"
|
|
|
|
#include "PresetBundle.hpp"
|
|
|
|
#include "libslic3r/Model.hpp"
|
|
|
|
|
|
|
|
#include <boost/algorithm/string.hpp>
|
|
|
|
|
|
|
|
#include "I18N.hpp"
|
|
|
|
|
|
|
|
#include <wx/wupdlock.h>
|
|
|
|
|
|
|
|
namespace Slic3r
|
|
|
|
{
|
|
|
|
namespace GUI
|
2019-05-30 10:41:16 +00:00
|
|
|
{
|
|
|
|
|
2019-05-28 14:38:04 +00:00
|
|
|
ObjectLayers::ObjectLayers(wxWindow* parent) :
|
|
|
|
OG_Settings(parent, true)
|
|
|
|
{
|
2019-06-04 09:51:25 +00:00
|
|
|
m_grid_sizer = new wxFlexGridSizer(3, 5, 5); // "Min Z", "Max Z", "Layer height" & buttons sizer
|
|
|
|
m_grid_sizer->SetFlexibleDirection(wxHORIZONTAL);
|
2019-06-03 13:35:21 +00:00
|
|
|
|
2019-05-28 14:38:04 +00:00
|
|
|
// Legend for object layers
|
2019-06-03 13:35:21 +00:00
|
|
|
for (const std::string col : { "Min Z", "Max Z", "Layer height" }) {
|
2019-06-04 09:51:25 +00:00
|
|
|
auto temp = new wxStaticText(m_parent, wxID_ANY, _(L(col)), wxDefaultPosition, /*size*/wxDefaultSize, wxST_ELLIPSIZE_MIDDLE);
|
|
|
|
temp->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
|
|
|
temp->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
|
|
|
temp->SetFont(wxGetApp().bold_font());
|
2019-05-30 10:41:16 +00:00
|
|
|
|
2019-06-04 09:51:25 +00:00
|
|
|
m_grid_sizer->Add(temp);
|
2019-05-28 14:38:04 +00:00
|
|
|
}
|
|
|
|
|
2019-06-04 09:51:25 +00:00
|
|
|
m_og->sizer->Clear(true);
|
|
|
|
m_og->sizer->Add(m_grid_sizer, 0, wxEXPAND | wxALL, wxOSX ? 0 : 5);
|
2019-05-28 14:38:04 +00:00
|
|
|
|
2019-05-30 14:53:17 +00:00
|
|
|
m_bmp_delete = ScalableBitmap(parent, "remove_copies"/*"cross"*/);
|
2019-05-28 14:38:04 +00:00
|
|
|
m_bmp_add = ScalableBitmap(parent, "add_copies");
|
|
|
|
}
|
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
void ObjectLayers::select_editor(LayerRangeEditor* editor, const bool is_last_edited_range)
|
|
|
|
{
|
|
|
|
if (is_last_edited_range && m_selection_type == editor->type()) {
|
|
|
|
editor->SetFocus();
|
|
|
|
editor->SetInsertionPointEnd();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-06 12:14:29 +00:00
|
|
|
wxSizer* ObjectLayers::create_layer(const t_layer_height_range& range)
|
2019-05-30 10:41:16 +00:00
|
|
|
{
|
2019-06-06 12:14:29 +00:00
|
|
|
const bool is_last_edited_range = range == m_last_edited_range;
|
2019-06-05 09:03:46 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
auto set_focus_fn = [range, this](const EditorType type)
|
|
|
|
{
|
|
|
|
m_last_edited_range = range;
|
|
|
|
m_selection_type = type;
|
|
|
|
};
|
|
|
|
|
2019-06-04 09:51:25 +00:00
|
|
|
// Add control for the "Min Z"
|
2019-06-05 09:03:46 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
auto editor = new LayerRangeEditor(m_parent, double_to_string(range.first), etMinZ,
|
|
|
|
set_focus_fn, [range, this](coordf_t min_z, bool enter_pressed)
|
2019-06-05 09:03:46 +00:00
|
|
|
{
|
2019-06-12 14:28:25 +00:00
|
|
|
if (fabs(min_z - range.first) < EPSILON || min_z > range.second) {
|
|
|
|
m_selection_type = etUndef;
|
|
|
|
return false;
|
2019-06-05 09:03:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// data for next focusing
|
2019-06-12 14:28:25 +00:00
|
|
|
const t_layer_height_range& new_range = { min_z, range.second };
|
|
|
|
if (enter_pressed) {
|
|
|
|
m_last_edited_range = new_range;
|
|
|
|
m_selection_type = etMinZ;
|
|
|
|
}
|
2019-06-05 09:03:46 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
return wxGetApp().obj_list()->edit_layer_range(range, new_range);
|
2019-06-06 12:14:29 +00:00
|
|
|
});
|
2019-06-05 09:03:46 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
select_editor(editor, is_last_edited_range);
|
|
|
|
m_grid_sizer->Add(editor);
|
2019-06-04 09:51:25 +00:00
|
|
|
|
|
|
|
// Add control for the "Max Z"
|
2019-06-05 09:03:46 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
editor = new LayerRangeEditor(m_parent, double_to_string(range.second), etMaxZ,
|
|
|
|
set_focus_fn, [range, this](coordf_t max_z, bool enter_pressed)
|
2019-06-05 09:03:46 +00:00
|
|
|
{
|
2019-06-12 14:28:25 +00:00
|
|
|
if (fabs(max_z - range.second) < EPSILON || range.first > max_z) {
|
|
|
|
m_selection_type = etUndef;
|
2019-06-05 09:03:46 +00:00
|
|
|
return false; // LayersList would not be updated/recreated
|
|
|
|
}
|
|
|
|
|
|
|
|
// data for next focusing
|
2019-06-12 14:28:25 +00:00
|
|
|
const t_layer_height_range& new_range = { range.first, max_z };
|
|
|
|
if (enter_pressed) {
|
|
|
|
m_last_edited_range = new_range;
|
|
|
|
m_selection_type = etMaxZ;
|
|
|
|
}
|
2019-06-05 09:03:46 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
return wxGetApp().obj_list()->edit_layer_range(range, new_range);
|
2019-06-04 15:30:44 +00:00
|
|
|
});
|
2019-06-05 09:03:46 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
select_editor(editor, is_last_edited_range);
|
|
|
|
m_grid_sizer->Add(editor);
|
2019-06-04 09:51:25 +00:00
|
|
|
|
|
|
|
// Add control for the "Layer height"
|
2019-06-05 09:03:46 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
editor = new LayerRangeEditor(m_parent,
|
2019-06-06 12:14:29 +00:00
|
|
|
double_to_string(m_object->layer_config_ranges[range].option("layer_height")->getFloat()),
|
2019-06-12 14:28:25 +00:00
|
|
|
etLayerHeight, set_focus_fn, [range, this](coordf_t layer_height, bool)
|
2019-06-05 09:03:46 +00:00
|
|
|
{
|
2019-06-12 14:28:25 +00:00
|
|
|
return wxGetApp().obj_list()->edit_layer_range(range, layer_height);
|
2019-06-05 09:03:46 +00:00
|
|
|
});
|
2019-06-04 09:51:25 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
select_editor(editor, is_last_edited_range);
|
|
|
|
|
2019-06-06 12:14:29 +00:00
|
|
|
auto sizer = new wxBoxSizer(wxHORIZONTAL);
|
2019-06-12 14:28:25 +00:00
|
|
|
sizer->Add(editor);
|
2019-06-04 09:51:25 +00:00
|
|
|
m_grid_sizer->Add(sizer);
|
|
|
|
|
|
|
|
return sizer;
|
2019-05-30 10:41:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ObjectLayers::create_layers_list()
|
|
|
|
{
|
2019-06-03 13:35:21 +00:00
|
|
|
for (const auto layer : m_object->layer_config_ranges)
|
2019-05-31 08:54:52 +00:00
|
|
|
{
|
2019-06-06 12:14:29 +00:00
|
|
|
const t_layer_height_range& range = layer.first;
|
|
|
|
auto sizer = create_layer(range);
|
2019-05-30 10:41:16 +00:00
|
|
|
|
2019-06-06 12:14:29 +00:00
|
|
|
auto del_btn = new ScalableButton(m_parent, wxID_ANY, m_bmp_delete);
|
2019-06-04 09:51:25 +00:00
|
|
|
del_btn->SetToolTip(_(L("Remove layer")));
|
2019-05-30 10:41:16 +00:00
|
|
|
|
2019-06-06 12:14:29 +00:00
|
|
|
sizer->Add(del_btn, 0, wxRIGHT | wxLEFT, em_unit(m_parent));
|
2019-05-30 10:41:16 +00:00
|
|
|
|
2019-06-06 12:14:29 +00:00
|
|
|
del_btn->Bind(wxEVT_BUTTON, [this, range](wxEvent &event) {
|
|
|
|
wxGetApp().obj_list()->del_layer_range(range);
|
2019-06-04 09:51:25 +00:00
|
|
|
});
|
2019-05-30 10:41:16 +00:00
|
|
|
|
2019-06-06 12:14:29 +00:00
|
|
|
auto add_btn = new ScalableButton(m_parent, wxID_ANY, m_bmp_add);
|
2019-06-04 09:51:25 +00:00
|
|
|
add_btn->SetToolTip(_(L("Add layer")));
|
2019-05-30 10:41:16 +00:00
|
|
|
|
2019-06-06 12:14:29 +00:00
|
|
|
sizer->Add(add_btn, 0, wxRIGHT, em_unit(m_parent));
|
2019-05-30 10:41:16 +00:00
|
|
|
|
2019-06-06 12:14:29 +00:00
|
|
|
add_btn->Bind(wxEVT_BUTTON, [this, range](wxEvent &event) {
|
|
|
|
wxGetApp().obj_list()->add_layer_range_after_current(range);
|
2019-06-04 09:51:25 +00:00
|
|
|
});
|
2019-05-30 10:41:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-28 14:38:04 +00:00
|
|
|
void ObjectLayers::update_layers_list()
|
|
|
|
{
|
|
|
|
ObjectList* objects_ctrl = wxGetApp().obj_list();
|
|
|
|
if (objects_ctrl->multiple_selection()) return;
|
|
|
|
|
|
|
|
const auto item = objects_ctrl->GetSelection();
|
|
|
|
if (!item) return;
|
|
|
|
|
|
|
|
const int obj_idx = objects_ctrl->get_selected_obj_idx();
|
|
|
|
if (obj_idx < 0) return;
|
|
|
|
|
|
|
|
const ItemType type = objects_ctrl->GetModel()->GetItemType(item);
|
|
|
|
if (!(type & (itLayerRoot | itLayer))) return;
|
|
|
|
|
2019-05-30 10:41:16 +00:00
|
|
|
m_object = objects_ctrl->object(obj_idx);
|
2019-06-03 13:35:21 +00:00
|
|
|
if (!m_object || m_object->layer_config_ranges.empty()) return;
|
2019-05-28 14:38:04 +00:00
|
|
|
|
2019-05-30 10:41:16 +00:00
|
|
|
// Delete all controls from options group except of the legends
|
2019-05-28 14:38:04 +00:00
|
|
|
|
2019-06-04 09:51:25 +00:00
|
|
|
const int cols = m_grid_sizer->GetEffectiveColsCount();
|
|
|
|
const int rows = m_grid_sizer->GetEffectiveRowsCount();
|
2019-05-28 14:38:04 +00:00
|
|
|
for (int idx = cols*rows-1; idx >= cols; idx--) {
|
2019-06-04 09:51:25 +00:00
|
|
|
wxSizerItem* t = m_grid_sizer->GetItem(idx);
|
2019-05-30 10:41:16 +00:00
|
|
|
if (t->IsSizer())
|
|
|
|
t->GetSizer()->Clear(true);
|
2019-06-04 09:51:25 +00:00
|
|
|
else
|
|
|
|
t->DeleteWindows();
|
|
|
|
m_grid_sizer->Remove(idx);
|
2019-05-28 14:38:04 +00:00
|
|
|
}
|
|
|
|
|
2019-05-30 10:41:16 +00:00
|
|
|
// Add new control according to the selected item
|
|
|
|
|
|
|
|
if (type & itLayerRoot)
|
|
|
|
create_layers_list();
|
|
|
|
else
|
2019-06-06 12:14:29 +00:00
|
|
|
create_layer(objects_ctrl->GetModel()->GetLayerRangeByItem(item));
|
2019-05-30 10:41:16 +00:00
|
|
|
|
|
|
|
m_parent->Layout();
|
2019-05-28 14:38:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void ObjectLayers::UpdateAndShow(const bool show)
|
|
|
|
{
|
|
|
|
if (show)
|
|
|
|
update_layers_list();
|
|
|
|
|
|
|
|
OG_Settings::UpdateAndShow(show);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ObjectLayers::msw_rescale()
|
|
|
|
{
|
|
|
|
m_bmp_delete.msw_rescale();
|
|
|
|
m_bmp_add.msw_rescale();
|
|
|
|
}
|
|
|
|
|
2019-06-04 13:22:29 +00:00
|
|
|
LayerRangeEditor::LayerRangeEditor( wxWindow* parent,
|
|
|
|
const wxString& value,
|
2019-06-12 14:28:25 +00:00
|
|
|
EditorType type,
|
|
|
|
std::function<void(EditorType)> set_focus_fn,
|
|
|
|
std::function<bool(coordf_t, bool enter_pressed)> edit_fn
|
2019-06-04 13:22:29 +00:00
|
|
|
) :
|
2019-06-12 14:28:25 +00:00
|
|
|
m_valid_value(value),
|
|
|
|
m_type(type),
|
2019-06-04 15:30:44 +00:00
|
|
|
wxTextCtrl(parent, wxID_ANY, value, wxDefaultPosition,
|
2019-06-06 12:14:29 +00:00
|
|
|
wxSize(8 * em_unit(parent), wxDefaultCoord), wxTE_PROCESS_ENTER)
|
2019-06-04 13:22:29 +00:00
|
|
|
{
|
|
|
|
this->SetFont(wxGetApp().normal_font());
|
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
this->Bind(wxEVT_TEXT_ENTER, [this, edit_fn](wxEvent&)
|
2019-06-04 13:22:29 +00:00
|
|
|
{
|
2019-06-05 09:03:46 +00:00
|
|
|
m_enter_pressed = true;
|
|
|
|
// If LayersList wasn't updated/recreated, we can call wxEVT_KILL_FOCUS.Skip()
|
2019-06-12 14:28:25 +00:00
|
|
|
if (m_type&etLayerHeight) {
|
|
|
|
if (!edit_fn(get_value(), true))
|
|
|
|
SetValue(m_valid_value);
|
|
|
|
else
|
|
|
|
m_valid_value = double_to_string(get_value());
|
2019-06-05 09:03:46 +00:00
|
|
|
m_call_kill_focus = true;
|
2019-06-12 14:28:25 +00:00
|
|
|
}
|
|
|
|
else if (!edit_fn(get_value(), true)) {
|
|
|
|
SetValue(m_valid_value);
|
|
|
|
m_call_kill_focus = true;
|
|
|
|
}
|
|
|
|
}, this->GetId());
|
2019-06-04 13:22:29 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
this->Bind(wxEVT_KILL_FOCUS, [this, edit_fn](wxFocusEvent& e)
|
2019-06-04 13:22:29 +00:00
|
|
|
{
|
2019-06-04 15:30:44 +00:00
|
|
|
if (!m_enter_pressed) {
|
2019-06-05 09:03:46 +00:00
|
|
|
// If LayersList wasn't updated/recreated, we should call e.Skip()
|
2019-06-12 14:28:25 +00:00
|
|
|
if (m_type & etLayerHeight) {
|
|
|
|
if (!edit_fn(get_value(), false))
|
|
|
|
SetValue(m_valid_value);
|
|
|
|
else
|
|
|
|
m_valid_value = double_to_string(get_value());
|
|
|
|
e.Skip();
|
|
|
|
}
|
|
|
|
else if (!edit_fn(get_value(), false)) {
|
|
|
|
SetValue(m_valid_value);
|
2019-06-05 09:03:46 +00:00
|
|
|
e.Skip();
|
2019-06-12 14:28:25 +00:00
|
|
|
}
|
2019-06-04 15:30:44 +00:00
|
|
|
}
|
2019-06-12 14:28:25 +00:00
|
|
|
else if (m_call_kill_focus) {
|
|
|
|
m_call_kill_focus = false;
|
2019-06-05 09:03:46 +00:00
|
|
|
e.Skip();
|
2019-06-12 14:28:25 +00:00
|
|
|
}
|
|
|
|
}, this->GetId());
|
2019-06-04 13:22:29 +00:00
|
|
|
|
2019-06-12 14:28:25 +00:00
|
|
|
this->Bind(wxEVT_LEFT_DOWN, ([this, set_focus_fn](wxEvent& e)
|
|
|
|
{
|
|
|
|
set_focus_fn(m_type);
|
|
|
|
e.Skip();
|
|
|
|
}));
|
2019-06-04 13:22:29 +00:00
|
|
|
|
|
|
|
this->Bind(wxEVT_CHAR, ([this](wxKeyEvent& event)
|
|
|
|
{
|
|
|
|
// select all text using Ctrl+A
|
|
|
|
if (wxGetKeyState(wxKeyCode('A')) && wxGetKeyState(WXK_CONTROL))
|
|
|
|
this->SetSelection(-1, -1); //select all
|
|
|
|
event.Skip();
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
coordf_t LayerRangeEditor::get_value()
|
|
|
|
{
|
|
|
|
wxString str = GetValue();
|
|
|
|
|
|
|
|
coordf_t layer_height;
|
|
|
|
// Replace the first occurence of comma in decimal number.
|
|
|
|
str.Replace(",", ".", false);
|
|
|
|
if (str == ".")
|
|
|
|
layer_height = 0.0;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!str.ToCDouble(&layer_height) || layer_height < 0.0f)
|
|
|
|
{
|
|
|
|
show_error(m_parent, _(L("Invalid numeric input.")));
|
|
|
|
SetValue(double_to_string(layer_height));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return layer_height;
|
|
|
|
}
|
|
|
|
|
2019-05-28 14:38:04 +00:00
|
|
|
} //namespace GUI
|
|
|
|
} //namespace Slic3r
|