WIP: World / local transformations of an object instance from the side panel.
This commit is contained in:
parent
5d2537af35
commit
2cc7b00a7d
7 changed files with 196 additions and 85 deletions
|
@ -275,6 +275,8 @@ public:
|
||||||
int volume_id;
|
int volume_id;
|
||||||
// Instance ID, which is equal to the index of the respective ModelInstance in ModelObject.instances array.
|
// Instance ID, which is equal to the index of the respective ModelInstance in ModelObject.instances array.
|
||||||
int instance_id;
|
int instance_id;
|
||||||
|
bool operator==(const CompositeID &rhs) const { return object_id == rhs.object_id && volume_id == rhs.volume_id && instance_id == rhs.instance_id; }
|
||||||
|
bool operator!=(const CompositeID &rhs) const { return ! (*this == rhs); }
|
||||||
};
|
};
|
||||||
CompositeID composite_id;
|
CompositeID composite_id;
|
||||||
// Fingerprint of the source geometry. For ModelVolumes, it is the ModelVolume::ID and ModelInstanceID,
|
// Fingerprint of the source geometry. For ModelVolumes, it is the ModelVolume::ID and ModelInstanceID,
|
||||||
|
|
|
@ -1870,8 +1870,12 @@ void GLCanvas3D::reload_scene(bool refresh_immediately, bool force_full_scene_re
|
||||||
if (m_reload_delayed)
|
if (m_reload_delayed)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
bool update_object_list = false;
|
||||||
|
|
||||||
if (m_regenerate_volumes)
|
if (m_regenerate_volumes)
|
||||||
{
|
{
|
||||||
|
if (m_volumes.volumes != glvolumes_new)
|
||||||
|
update_object_list = true;
|
||||||
m_volumes.volumes = std::move(glvolumes_new);
|
m_volumes.volumes = std::move(glvolumes_new);
|
||||||
for (unsigned int obj_idx = 0; obj_idx < (unsigned int)m_model->objects.size(); ++ obj_idx) {
|
for (unsigned int obj_idx = 0; obj_idx < (unsigned int)m_model->objects.size(); ++ obj_idx) {
|
||||||
const ModelObject &model_object = *m_model->objects[obj_idx];
|
const ModelObject &model_object = *m_model->objects[obj_idx];
|
||||||
|
@ -1886,12 +1890,16 @@ void GLCanvas3D::reload_scene(bool refresh_immediately, bool force_full_scene_re
|
||||||
// New volume.
|
// New volume.
|
||||||
m_volumes.load_object_volume(&model_object, obj_idx, volume_idx, instance_idx, m_color_by, m_use_VBOs && m_initialized);
|
m_volumes.load_object_volume(&model_object, obj_idx, volume_idx, instance_idx, m_color_by, m_use_VBOs && m_initialized);
|
||||||
m_volumes.volumes.back()->geometry_id = key.geometry_id;
|
m_volumes.volumes.back()->geometry_id = key.geometry_id;
|
||||||
|
update_object_list = true;
|
||||||
} else {
|
} else {
|
||||||
// Recycling an old GLVolume.
|
// Recycling an old GLVolume.
|
||||||
GLVolume &existing_volume = *m_volumes.volumes[it->volume_idx];
|
GLVolume &existing_volume = *m_volumes.volumes[it->volume_idx];
|
||||||
assert(existing_volume.geometry_id == key.geometry_id);
|
assert(existing_volume.geometry_id == key.geometry_id);
|
||||||
// Update the Object/Volume/Instance indices into the current Model.
|
// Update the Object/Volume/Instance indices into the current Model.
|
||||||
existing_volume.composite_id = it->composite_id;
|
if (existing_volume.composite_id != it->composite_id) {
|
||||||
|
existing_volume.composite_id = it->composite_id;
|
||||||
|
update_object_list = true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1999,7 +2007,8 @@ void GLCanvas3D::reload_scene(bool refresh_immediately, bool force_full_scene_re
|
||||||
m_gizmos.update_data(*this);
|
m_gizmos.update_data(*this);
|
||||||
|
|
||||||
// Update the toolbar
|
// Update the toolbar
|
||||||
post_event(SimpleEvent(EVT_GLCANVAS_OBJECT_SELECT));
|
if (update_object_list)
|
||||||
|
post_event(SimpleEvent(EVT_GLCANVAS_OBJECT_SELECT));
|
||||||
|
|
||||||
// checks for geometry outside the print volume to render it accordingly
|
// checks for geometry outside the print volume to render it accordingly
|
||||||
if (!m_volumes.empty())
|
if (!m_volumes.empty())
|
||||||
|
|
|
@ -16,6 +16,50 @@ namespace Slic3r
|
||||||
namespace GUI
|
namespace GUI
|
||||||
{
|
{
|
||||||
|
|
||||||
|
static wxBitmapComboBox* create_word_local_combo(wxWindow *parent)
|
||||||
|
{
|
||||||
|
wxSize size(15 * wxGetApp().em_unit(), -1);
|
||||||
|
|
||||||
|
wxBitmapComboBox *temp = nullptr;
|
||||||
|
#ifdef __WXOSX__
|
||||||
|
/* wxBitmapComboBox with wxCB_READONLY style return NULL for GetTextCtrl(),
|
||||||
|
* so ToolTip doesn't shown.
|
||||||
|
* Next workaround helps to solve this problem
|
||||||
|
*/
|
||||||
|
temp = new wxBitmapComboBox();
|
||||||
|
temp->SetTextCtrlStyle(wxTE_READONLY);
|
||||||
|
temp->Create(parent, wxID_ANY, wxString(""), wxDefaultPosition, size, 0, nullptr);
|
||||||
|
#else
|
||||||
|
temp = new wxBitmapComboBox(parent, wxID_ANY, wxString(""), wxDefaultPosition, size, 0, nullptr, wxCB_READONLY);
|
||||||
|
#endif //__WXOSX__
|
||||||
|
|
||||||
|
temp->SetFont(Slic3r::GUI::wxGetApp().normal_font());
|
||||||
|
temp->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
||||||
|
|
||||||
|
temp->Append(_(L("World")));
|
||||||
|
temp->Append(_(L("Local")));
|
||||||
|
temp->SetSelection(0);
|
||||||
|
temp->SetValue(temp->GetString(0));
|
||||||
|
|
||||||
|
#ifndef __WXGTK__
|
||||||
|
/* Workaround for a correct rendering of the control without Bitmap (under MSW and OSX):
|
||||||
|
*
|
||||||
|
* 1. We should create small Bitmap to fill Bitmaps RefData,
|
||||||
|
* ! in this case wxBitmap.IsOK() return true.
|
||||||
|
* 2. But then set width to 0 value for no using of bitmap left and right spacing
|
||||||
|
* 3. Set this empty bitmap to the at list one item and BitmapCombobox will be recreated correct
|
||||||
|
*
|
||||||
|
* Note: Set bitmap height to the Font size because of OSX rendering.
|
||||||
|
*/
|
||||||
|
wxBitmap empty_bmp(1, temp->GetFont().GetPixelSize().y + 2);
|
||||||
|
empty_bmp.SetWidth(0);
|
||||||
|
temp->SetItemBitmap(0, empty_bmp);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
temp->SetToolTip(_(L("Select coordinate space, in which the transformation will be performed.")));
|
||||||
|
return temp;
|
||||||
|
}
|
||||||
|
|
||||||
ObjectManipulation::ObjectManipulation(wxWindow* parent) :
|
ObjectManipulation::ObjectManipulation(wxWindow* parent) :
|
||||||
OG_Settings(parent, true)
|
OG_Settings(parent, true)
|
||||||
#ifndef __APPLE__
|
#ifndef __APPLE__
|
||||||
|
@ -63,6 +107,12 @@ ObjectManipulation::ObjectManipulation(wxWindow* parent) :
|
||||||
Option option = Option(def, axis + "_axis_legend");
|
Option option = Option(def, axis + "_axis_legend");
|
||||||
line.append_option(option);
|
line.append_option(option);
|
||||||
}
|
}
|
||||||
|
line.near_label_widget = [this](wxWindow* parent) {
|
||||||
|
wxBitmapComboBox *combo = create_word_local_combo(parent);
|
||||||
|
combo->Bind(wxEVT_COMBOBOX, ([this](wxCommandEvent &evt) { this->set_world_coordinates(evt.GetSelection() != 1); }), combo->GetId());
|
||||||
|
m_word_local_combo = combo;
|
||||||
|
return combo;
|
||||||
|
};
|
||||||
m_og->append_line(line);
|
m_og->append_line(line);
|
||||||
|
|
||||||
auto add_og_to_object_settings = [this, field_width](const std::string& option_name, const std::string& sidetext)
|
auto add_og_to_object_settings = [this, field_width](const std::string& option_name, const std::string& sidetext)
|
||||||
|
@ -145,9 +195,8 @@ bool ObjectManipulation::IsShown()
|
||||||
|
|
||||||
void ObjectManipulation::UpdateAndShow(const bool show)
|
void ObjectManipulation::UpdateAndShow(const bool show)
|
||||||
{
|
{
|
||||||
if (show) {
|
if (show)
|
||||||
update_settings_value(wxGetApp().plater()->canvas3D()->get_selection());
|
update_settings_value(wxGetApp().plater()->canvas3D()->get_selection());
|
||||||
}
|
|
||||||
|
|
||||||
OG_Settings::UpdateAndShow(show);
|
OG_Settings::UpdateAndShow(show);
|
||||||
}
|
}
|
||||||
|
@ -164,8 +213,8 @@ void ObjectManipulation::update_settings_value(const Selection& selection)
|
||||||
// all volumes in the selection belongs to the same instance, any of them contains the needed instance data, so we take the first one
|
// all volumes in the selection belongs to the same instance, any of them contains the needed instance data, so we take the first one
|
||||||
const GLVolume* volume = selection.get_volume(*selection.get_volume_idxs().begin());
|
const GLVolume* volume = selection.get_volume(*selection.get_volume_idxs().begin());
|
||||||
m_new_position = volume->get_instance_offset();
|
m_new_position = volume->get_instance_offset();
|
||||||
m_new_rotation = volume->get_instance_rotation();
|
m_new_rotation = volume->get_instance_rotation() * (180. / M_PI);
|
||||||
m_new_scale = volume->get_instance_scaling_factor();
|
m_new_scale = volume->get_instance_scaling_factor() * 100.;
|
||||||
int obj_idx = volume->object_idx();
|
int obj_idx = volume->object_idx();
|
||||||
int instance_idx = volume->instance_idx();
|
int instance_idx = volume->instance_idx();
|
||||||
if ((0 <= obj_idx) && (obj_idx < (int)wxGetApp().model_objects()->size()))
|
if ((0 <= obj_idx) && (obj_idx < (int)wxGetApp().model_objects()->size()))
|
||||||
|
@ -178,7 +227,7 @@ void ObjectManipulation::update_settings_value(const Selection& selection)
|
||||||
changed_box = true;
|
changed_box = true;
|
||||||
}
|
}
|
||||||
//FIXME matching an instance idx may not be enough. Check for ModelObject id an all ModelVolume ids.
|
//FIXME matching an instance idx may not be enough. Check for ModelObject id an all ModelVolume ids.
|
||||||
if (changed_box || !m_cache.instance.matches_instance(instance_idx) || !m_cache.scale.isApprox(100.0 * m_new_scale))
|
if (changed_box || !m_cache.instance.matches_instance(instance_idx) || !m_cache.scale.isApprox(m_new_scale))
|
||||||
m_new_size = volume->get_instance_transformation().get_scaling_factor().cwiseProduct(m_cache.instance.box_size);
|
m_new_size = volume->get_instance_transformation().get_scaling_factor().cwiseProduct(m_cache.instance.box_size);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -197,7 +246,7 @@ void ObjectManipulation::update_settings_value(const Selection& selection)
|
||||||
const BoundingBoxf3& box = selection.get_bounding_box();
|
const BoundingBoxf3& box = selection.get_bounding_box();
|
||||||
m_new_position = box.center();
|
m_new_position = box.center();
|
||||||
m_new_rotation = Vec3d::Zero();
|
m_new_rotation = Vec3d::Zero();
|
||||||
m_new_scale = Vec3d(1.0, 1.0, 1.0);
|
m_new_scale = Vec3d(100., 100., 100.);
|
||||||
m_new_size = box.size();
|
m_new_size = box.size();
|
||||||
m_new_rotate_label_string = L("Rotate");
|
m_new_rotate_label_string = L("Rotate");
|
||||||
m_new_scale_label_string = L("Scale");
|
m_new_scale_label_string = L("Scale");
|
||||||
|
@ -210,8 +259,8 @@ void ObjectManipulation::update_settings_value(const Selection& selection)
|
||||||
// the selection contains a single volume
|
// the selection contains a single volume
|
||||||
const GLVolume* volume = selection.get_volume(*selection.get_volume_idxs().begin());
|
const GLVolume* volume = selection.get_volume(*selection.get_volume_idxs().begin());
|
||||||
m_new_position = volume->get_volume_offset();
|
m_new_position = volume->get_volume_offset();
|
||||||
m_new_rotation = volume->get_volume_rotation();
|
m_new_rotation = volume->get_volume_rotation() * (180. / M_PI);
|
||||||
m_new_scale = volume->get_volume_scaling_factor();
|
m_new_scale = volume->get_volume_scaling_factor() * 100.;
|
||||||
m_new_size = volume->get_volume_transformation().get_scaling_factor().cwiseProduct(volume->bounding_box.size());
|
m_new_size = volume->get_volume_transformation().get_scaling_factor().cwiseProduct(volume->bounding_box.size());
|
||||||
m_new_enabled = true;
|
m_new_enabled = true;
|
||||||
}
|
}
|
||||||
|
@ -226,7 +275,7 @@ void ObjectManipulation::update_settings_value(const Selection& selection)
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// No selection, reset the cache.
|
// No selection, reset the cache.
|
||||||
assert(selection.is_empty());
|
// assert(selection.is_empty());
|
||||||
reset_settings_value();
|
reset_settings_value();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -249,26 +298,24 @@ void ObjectManipulation::update_if_dirty()
|
||||||
update_label(m_cache.rotate_label_string, m_new_rotate_label_string, m_rotate_Label);
|
update_label(m_cache.rotate_label_string, m_new_rotate_label_string, m_rotate_Label);
|
||||||
update_label(m_cache.scale_label_string, m_new_scale_label_string, m_scale_Label);
|
update_label(m_cache.scale_label_string, m_new_scale_label_string, m_scale_Label);
|
||||||
|
|
||||||
Vec3d scale = m_new_scale * 100.0;
|
|
||||||
Vec3d deg_rotation = (180.0 / M_PI) * m_new_rotation;
|
|
||||||
|
|
||||||
char axis[2] = "x";
|
char axis[2] = "x";
|
||||||
for (int i = 0; i < 3; ++ i, ++ axis[0]) {
|
for (int i = 0; i < 3; ++ i, ++ axis[0]) {
|
||||||
if (m_cache.position(i) != m_new_position(i))
|
auto update = [this, i, &axis](Vec3d &cached, Vec3d &cached_rounded, const char *key, const Vec3d &new_value) {
|
||||||
m_og->set_value(std::string("position_") + axis, double_to_string(m_new_position(i), 2));
|
wxString new_text = double_to_string(new_value(i), 2);
|
||||||
if (m_cache.scale(i) != scale(i))
|
double new_rounded;
|
||||||
m_og->set_value(std::string("scale_") + axis, double_to_string(scale(i), 2));
|
new_text.ToDouble(&new_rounded);
|
||||||
if (m_cache.size(i) != m_new_size(i))
|
if (std::abs(cached_rounded(i) - new_rounded) > EPSILON) {
|
||||||
m_og->set_value(std::string("size_") + axis, double_to_string(m_new_size(i), 2));
|
cached_rounded(i) = new_rounded;
|
||||||
if (m_cache.rotation(i) != m_new_rotation(i) || m_new_rotation(i) == 0.0)
|
m_og->set_value(std::string(key) + axis, new_text);
|
||||||
m_og->set_value(std::string("rotation_") + axis, double_to_string(deg_rotation(i), 2));
|
}
|
||||||
|
cached(i) = new_value(i);
|
||||||
|
};
|
||||||
|
update(m_cache.position, m_cache.position_rounded, "position_", m_new_position);
|
||||||
|
update(m_cache.scale, m_cache.scale_rounded, "scale_", m_new_scale);
|
||||||
|
update(m_cache.size, m_cache.size_rounded, "size_", m_new_size);
|
||||||
|
update(m_cache.rotation, m_cache.rotation_rounded, "rotation_", m_new_rotation);
|
||||||
}
|
}
|
||||||
|
|
||||||
m_cache.position = m_new_position;
|
|
||||||
m_cache.scale = scale;
|
|
||||||
m_cache.size = m_new_size;
|
|
||||||
m_cache.rotation = deg_rotation;
|
|
||||||
|
|
||||||
if (wxGetApp().plater()->canvas3D()->get_selection().requires_uniform_scale()) {
|
if (wxGetApp().plater()->canvas3D()->get_selection().requires_uniform_scale()) {
|
||||||
m_lock_bnt->SetLock(true);
|
m_lock_bnt->SetLock(true);
|
||||||
m_lock_bnt->Disable();
|
m_lock_bnt->Disable();
|
||||||
|
@ -278,6 +325,12 @@ void ObjectManipulation::update_if_dirty()
|
||||||
m_lock_bnt->Enable();
|
m_lock_bnt->Enable();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
int new_selection = m_world_coordinates ? 0 : 1;
|
||||||
|
if (m_word_local_combo->GetSelection() != new_selection)
|
||||||
|
m_word_local_combo->SetSelection(new_selection);
|
||||||
|
}
|
||||||
|
|
||||||
if (m_new_enabled)
|
if (m_new_enabled)
|
||||||
m_og->enable();
|
m_og->enable();
|
||||||
else
|
else
|
||||||
|
@ -314,8 +367,14 @@ void ObjectManipulation::reset_settings_value()
|
||||||
m_dirty = true;
|
m_dirty = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ObjectManipulation::change_position_value(const Vec3d& position)
|
void ObjectManipulation::change_position_value(int axis, double value)
|
||||||
{
|
{
|
||||||
|
if (std::abs(m_cache.position_rounded(axis) - value) < EPSILON)
|
||||||
|
return;
|
||||||
|
|
||||||
|
Vec3d position = m_cache.position;
|
||||||
|
position(axis) = value;
|
||||||
|
|
||||||
auto canvas = wxGetApp().plater()->canvas3D();
|
auto canvas = wxGetApp().plater()->canvas3D();
|
||||||
Selection& selection = canvas->get_selection();
|
Selection& selection = canvas->get_selection();
|
||||||
selection.start_dragging();
|
selection.start_dragging();
|
||||||
|
@ -323,10 +382,18 @@ void ObjectManipulation::change_position_value(const Vec3d& position)
|
||||||
canvas->do_move();
|
canvas->do_move();
|
||||||
|
|
||||||
m_cache.position = position;
|
m_cache.position = position;
|
||||||
|
m_cache.position_rounded(axis) = DBL_MAX;
|
||||||
|
this->UpdateAndShow(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ObjectManipulation::change_rotation_value(const Vec3d& rotation)
|
void ObjectManipulation::change_rotation_value(int axis, double value)
|
||||||
{
|
{
|
||||||
|
if (std::abs(m_cache.rotation_rounded(axis) - value) < EPSILON)
|
||||||
|
return;
|
||||||
|
|
||||||
|
Vec3d rotation = m_cache.rotation;
|
||||||
|
rotation(axis) = value;
|
||||||
|
|
||||||
GLCanvas3D* canvas = wxGetApp().plater()->canvas3D();
|
GLCanvas3D* canvas = wxGetApp().plater()->canvas3D();
|
||||||
Selection& selection = canvas->get_selection();
|
Selection& selection = canvas->get_selection();
|
||||||
|
|
||||||
|
@ -346,20 +413,36 @@ void ObjectManipulation::change_rotation_value(const Vec3d& rotation)
|
||||||
canvas->do_rotate();
|
canvas->do_rotate();
|
||||||
|
|
||||||
m_cache.rotation = rotation;
|
m_cache.rotation = rotation;
|
||||||
|
m_cache.rotation_rounded(axis) = DBL_MAX;
|
||||||
|
this->UpdateAndShow(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ObjectManipulation::change_scale_value(const Vec3d& scale)
|
void ObjectManipulation::change_scale_value(int axis, double value)
|
||||||
{
|
{
|
||||||
|
if (std::abs(m_cache.scale_rounded(axis) - value) < EPSILON)
|
||||||
|
return;
|
||||||
|
|
||||||
|
Vec3d scale = m_cache.scale;
|
||||||
|
scale(axis) = value;
|
||||||
|
|
||||||
this->do_scale(scale);
|
this->do_scale(scale);
|
||||||
|
|
||||||
if (!m_cache.scale.isApprox(scale))
|
if (!m_cache.scale.isApprox(scale))
|
||||||
m_cache.instance.instance_idx = -1;
|
m_cache.instance.instance_idx = -1;
|
||||||
|
|
||||||
m_cache.scale = scale;
|
m_cache.scale = scale;
|
||||||
|
m_cache.scale_rounded(axis) = DBL_MAX;
|
||||||
|
this->UpdateAndShow(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ObjectManipulation::change_size_value(const Vec3d& size)
|
void ObjectManipulation::change_size_value(int axis, double value)
|
||||||
{
|
{
|
||||||
|
if (std::abs(m_cache.size_rounded(axis) - value) < EPSILON)
|
||||||
|
return;
|
||||||
|
|
||||||
|
Vec3d size = m_cache.size;
|
||||||
|
size(axis) = value;
|
||||||
|
|
||||||
const Selection& selection = wxGetApp().plater()->canvas3D()->get_selection();
|
const Selection& selection = wxGetApp().plater()->canvas3D()->get_selection();
|
||||||
|
|
||||||
Vec3d ref_size = m_cache.size;
|
Vec3d ref_size = m_cache.size;
|
||||||
|
@ -371,9 +454,11 @@ void ObjectManipulation::change_size_value(const Vec3d& size)
|
||||||
else if (selection.is_single_full_instance() && ! m_world_coordinates)
|
else if (selection.is_single_full_instance() && ! m_world_coordinates)
|
||||||
ref_size = m_cache.instance.box_size;
|
ref_size = m_cache.instance.box_size;
|
||||||
|
|
||||||
this->do_scale(100.0 * Vec3d(size(0) / ref_size(0), size(1) / ref_size(1), size(2) / ref_size(2)));
|
this->do_scale(100. * Vec3d(size(0) / ref_size(0), size(1) / ref_size(1), size(2) / ref_size(2)));
|
||||||
|
|
||||||
m_cache.size = size;
|
m_cache.size = size;
|
||||||
|
m_cache.size_rounded(axis) = DBL_MAX;
|
||||||
|
this->UpdateAndShow(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ObjectManipulation::do_scale(const Vec3d &scale) const
|
void ObjectManipulation::do_scale(const Vec3d &scale) const
|
||||||
|
@ -407,20 +492,17 @@ void ObjectManipulation::on_change(t_config_option_key opt_key, const boost::any
|
||||||
if (!m_cache.is_valid())
|
if (!m_cache.is_valid())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Value of all three axes of the position / rotation / scale / size is extracted.
|
int axis = opt_key.back() - 'x';
|
||||||
Vec3d new_value;
|
double new_value = boost::any_cast<double>(m_og->get_value(opt_key));
|
||||||
opt_key.back() = 'x';
|
|
||||||
for (int i = 0; i < 3; ++ i, ++ opt_key.back())
|
|
||||||
new_value(i) = boost::any_cast<double>(m_og->get_value(opt_key));
|
|
||||||
|
|
||||||
if (boost::starts_with(opt_key, "position_"))
|
if (boost::starts_with(opt_key, "position_"))
|
||||||
change_position_value(new_value);
|
change_position_value(axis, new_value);
|
||||||
else if (boost::starts_with(opt_key, "rotation_"))
|
else if (boost::starts_with(opt_key, "rotation_"))
|
||||||
change_rotation_value(new_value);
|
change_rotation_value(axis, new_value);
|
||||||
else if (boost::starts_with(opt_key, "scale_"))
|
else if (boost::starts_with(opt_key, "scale_"))
|
||||||
change_scale_value(new_value);
|
change_scale_value(axis, new_value);
|
||||||
else if (boost::starts_with(opt_key, "size_"))
|
else if (boost::starts_with(opt_key, "size_"))
|
||||||
change_size_value(new_value);
|
change_size_value(axis, new_value);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ObjectManipulation::on_fill_empty_value(const std::string& opt_key)
|
void ObjectManipulation::on_fill_empty_value(const std::string& opt_key)
|
||||||
|
@ -435,19 +517,28 @@ void ObjectManipulation::on_fill_empty_value(const std::string& opt_key)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
const Vec3d *vec = nullptr;
|
const Vec3d *vec = nullptr;
|
||||||
if (boost::starts_with(opt_key, "position_"))
|
Vec3d *rounded = nullptr;
|
||||||
|
if (boost::starts_with(opt_key, "position_")) {
|
||||||
vec = &m_cache.position;
|
vec = &m_cache.position;
|
||||||
else if (boost::starts_with(opt_key, "rotation_"))
|
rounded = &m_cache.position_rounded;
|
||||||
|
} else if (boost::starts_with(opt_key, "rotation_")) {
|
||||||
vec = &m_cache.rotation;
|
vec = &m_cache.rotation;
|
||||||
else if (boost::starts_with(opt_key, "scale_"))
|
rounded = &m_cache.rotation_rounded;
|
||||||
|
} else if (boost::starts_with(opt_key, "scale_")) {
|
||||||
vec = &m_cache.scale;
|
vec = &m_cache.scale;
|
||||||
else if (boost::starts_with(opt_key, "size_"))
|
rounded = &m_cache.scale_rounded;
|
||||||
|
} else if (boost::starts_with(opt_key, "size_")) {
|
||||||
vec = &m_cache.size;
|
vec = &m_cache.size;
|
||||||
else
|
rounded = &m_cache.size_rounded;
|
||||||
|
} else
|
||||||
assert(false);
|
assert(false);
|
||||||
|
|
||||||
if (vec != nullptr)
|
if (vec != nullptr) {
|
||||||
m_og->set_value(opt_key, double_to_string((*vec)(opt_key.back() - 'x')));
|
int axis = opt_key.back() - 'x';
|
||||||
|
wxString new_text = double_to_string((*vec)(axis));
|
||||||
|
m_og->set_value(opt_key, new_text);
|
||||||
|
new_text.ToDouble(&(*rounded)(axis));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
} //namespace GUI
|
} //namespace GUI
|
||||||
|
|
|
@ -19,9 +19,13 @@ class ObjectManipulation : public OG_Settings
|
||||||
struct Cache
|
struct Cache
|
||||||
{
|
{
|
||||||
Vec3d position;
|
Vec3d position;
|
||||||
|
Vec3d position_rounded;
|
||||||
Vec3d rotation;
|
Vec3d rotation;
|
||||||
|
Vec3d rotation_rounded;
|
||||||
Vec3d scale;
|
Vec3d scale;
|
||||||
|
Vec3d scale_rounded;
|
||||||
Vec3d size;
|
Vec3d size;
|
||||||
|
Vec3d size_rounded;
|
||||||
|
|
||||||
std::string move_label_string;
|
std::string move_label_string;
|
||||||
std::string rotate_label_string;
|
std::string rotate_label_string;
|
||||||
|
@ -46,10 +50,10 @@ class ObjectManipulation : public OG_Settings
|
||||||
Cache() { reset(); }
|
Cache() { reset(); }
|
||||||
void reset()
|
void reset()
|
||||||
{
|
{
|
||||||
position = Vec3d(DBL_MAX, DBL_MAX, DBL_MAX);
|
position = position_rounded = Vec3d(DBL_MAX, DBL_MAX, DBL_MAX);
|
||||||
rotation = Vec3d(DBL_MAX, DBL_MAX, DBL_MAX);
|
rotation = rotation_rounded = Vec3d(DBL_MAX, DBL_MAX, DBL_MAX);
|
||||||
scale = Vec3d(DBL_MAX, DBL_MAX, DBL_MAX);
|
scale = scale_rounded = Vec3d(DBL_MAX, DBL_MAX, DBL_MAX);
|
||||||
size = Vec3d(DBL_MAX, DBL_MAX, DBL_MAX);
|
size = size_rounded = Vec3d(DBL_MAX, DBL_MAX, DBL_MAX);
|
||||||
move_label_string = "";
|
move_label_string = "";
|
||||||
rotate_label_string = "";
|
rotate_label_string = "";
|
||||||
scale_label_string = "";
|
scale_label_string = "";
|
||||||
|
@ -79,6 +83,7 @@ class ObjectManipulation : public OG_Settings
|
||||||
// Does the object manipulation panel work in World or Local coordinates?
|
// Does the object manipulation panel work in World or Local coordinates?
|
||||||
bool m_world_coordinates = true;
|
bool m_world_coordinates = true;
|
||||||
PrusaLockButton* m_lock_bnt{ nullptr };
|
PrusaLockButton* m_lock_bnt{ nullptr };
|
||||||
|
wxBitmapComboBox* m_word_local_combo = nullptr;
|
||||||
|
|
||||||
#ifndef __APPLE__
|
#ifndef __APPLE__
|
||||||
// Currently focused option name (empty if none)
|
// Currently focused option name (empty if none)
|
||||||
|
@ -101,6 +106,7 @@ public:
|
||||||
void set_uniform_scaling(const bool uniform_scale) { m_uniform_scale = uniform_scale;}
|
void set_uniform_scaling(const bool uniform_scale) { m_uniform_scale = uniform_scale;}
|
||||||
bool get_uniform_scaling() const { return m_uniform_scale; }
|
bool get_uniform_scaling() const { return m_uniform_scale; }
|
||||||
// Does the object manipulation panel work in World or Local coordinates?
|
// Does the object manipulation panel work in World or Local coordinates?
|
||||||
|
void set_world_coordinates(const bool world_coordinates) { m_world_coordinates = world_coordinates; this->UpdateAndShow(true); }
|
||||||
bool get_world_coordinates() const { return m_world_coordinates; }
|
bool get_world_coordinates() const { return m_world_coordinates; }
|
||||||
|
|
||||||
void reset_cache() { m_cache.reset(); }
|
void reset_cache() { m_cache.reset(); }
|
||||||
|
@ -120,10 +126,10 @@ private:
|
||||||
void update_rotation_value(const Vec3d& rotation);
|
void update_rotation_value(const Vec3d& rotation);
|
||||||
|
|
||||||
// change values
|
// change values
|
||||||
void change_position_value(const Vec3d& position);
|
void change_position_value(int axis, double value);
|
||||||
void change_rotation_value(const Vec3d& rotation);
|
void change_rotation_value(int axis, double value);
|
||||||
void change_scale_value(const Vec3d& scale);
|
void change_scale_value(int axis, double value);
|
||||||
void change_size_value(const Vec3d& size);
|
void change_size_value(int axis, double value);
|
||||||
void do_scale(const Vec3d &scale) const;
|
void do_scale(const Vec3d &scale) const;
|
||||||
|
|
||||||
void on_change(t_config_option_key opt_key, const boost::any& value);
|
void on_change(t_config_option_key opt_key, const boost::any& value);
|
||||||
|
|
|
@ -172,19 +172,21 @@ void OptionsGroup::append_line(const Line& line, wxStaticText** full_Label/* = n
|
||||||
// Build a label if we have it
|
// Build a label if we have it
|
||||||
wxStaticText* label=nullptr;
|
wxStaticText* label=nullptr;
|
||||||
if (label_width != 0) {
|
if (label_width != 0) {
|
||||||
long label_style = staticbox ? 0 : wxALIGN_RIGHT;
|
if (! line.near_label_widget || ! line.label.IsEmpty()) {
|
||||||
#ifdef __WXGTK__
|
long label_style = staticbox ? 0 : wxALIGN_RIGHT;
|
||||||
// workaround for correct text align of the StaticBox on Linux
|
#ifdef __WXGTK__
|
||||||
// flags wxALIGN_RIGHT and wxALIGN_CENTRE don't work when Ellipsize flags are _not_ given.
|
// workaround for correct text align of the StaticBox on Linux
|
||||||
// Text is properly aligned only when Ellipsize is checked.
|
// flags wxALIGN_RIGHT and wxALIGN_CENTRE don't work when Ellipsize flags are _not_ given.
|
||||||
label_style |= staticbox ? 0 : wxST_ELLIPSIZE_END;
|
// Text is properly aligned only when Ellipsize is checked.
|
||||||
#endif /* __WXGTK__ */
|
label_style |= staticbox ? 0 : wxST_ELLIPSIZE_END;
|
||||||
label = new wxStaticText(this->ctrl_parent(), wxID_ANY, line.label + (line.label.IsEmpty() ? "" : ": "),
|
#endif /* __WXGTK__ */
|
||||||
wxDefaultPosition, wxSize(label_width, -1), label_style);
|
label = new wxStaticText(this->ctrl_parent(), wxID_ANY, line.label + (line.label.IsEmpty() ? "" : ": "),
|
||||||
label->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
wxDefaultPosition, wxSize(label_width, -1), label_style);
|
||||||
label->SetFont(label_font);
|
label->SetBackgroundStyle(wxBG_STYLE_PAINT);
|
||||||
label->Wrap(label_width); // avoid a Linux/GTK bug
|
label->SetFont(label_font);
|
||||||
if (!line.near_label_widget)
|
label->Wrap(label_width); // avoid a Linux/GTK bug
|
||||||
|
}
|
||||||
|
if (! line.near_label_widget)
|
||||||
grid_sizer->Add(label, 0, (staticbox ? 0 : wxALIGN_RIGHT | wxRIGHT) | wxALIGN_CENTER_VERTICAL, line.label.IsEmpty() ? 0 : 5);
|
grid_sizer->Add(label, 0, (staticbox ? 0 : wxALIGN_RIGHT | wxRIGHT) | wxALIGN_CENTER_VERTICAL, line.label.IsEmpty() ? 0 : 5);
|
||||||
else if (line.near_label_widget && line.label.IsEmpty())
|
else if (line.near_label_widget && line.label.IsEmpty())
|
||||||
grid_sizer->Add(line.near_label_widget(this->ctrl_parent()), 0, wxRIGHT | wxALIGN_CENTER_VERTICAL, 7);
|
grid_sizer->Add(line.near_label_widget(this->ctrl_parent()), 0, wxRIGHT | wxALIGN_CENTER_VERTICAL, 7);
|
||||||
|
@ -196,7 +198,7 @@ void OptionsGroup::append_line(const Line& line, wxStaticText** full_Label/* = n
|
||||||
sizer->Add(line.near_label_widget(this->ctrl_parent()), 0, wxRIGHT, 7);
|
sizer->Add(line.near_label_widget(this->ctrl_parent()), 0, wxRIGHT, 7);
|
||||||
sizer->Add(label, 0, (staticbox ? 0 : wxALIGN_RIGHT | wxRIGHT) | wxALIGN_CENTER_VERTICAL, 5);
|
sizer->Add(label, 0, (staticbox ? 0 : wxALIGN_RIGHT | wxRIGHT) | wxALIGN_CENTER_VERTICAL, 5);
|
||||||
}
|
}
|
||||||
if (line.label_tooltip.compare("") != 0)
|
if (label != nullptr && line.label_tooltip != "")
|
||||||
label->SetToolTip(line.label_tooltip);
|
label->SetToolTip(line.label_tooltip);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1056,9 +1056,9 @@ void Sidebar::enable_buttons(bool enable)
|
||||||
p->btn_send_gcode->Enable(enable);
|
p->btn_send_gcode->Enable(enable);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Sidebar::show_reslice(bool show) const { p->btn_reslice->Show(show); }
|
bool Sidebar::show_reslice(bool show) const { return p->btn_reslice->Show(show); }
|
||||||
void Sidebar::show_export(bool show) const { p->btn_export_gcode->Show(show); }
|
bool Sidebar::show_export(bool show) const { return p->btn_export_gcode->Show(show); }
|
||||||
void Sidebar::show_send(bool show) const { p->btn_send_gcode->Show(show); }
|
bool Sidebar::show_send(bool show) const { return p->btn_send_gcode->Show(show); }
|
||||||
|
|
||||||
bool Sidebar::is_multifilament()
|
bool Sidebar::is_multifilament()
|
||||||
{
|
{
|
||||||
|
@ -3129,17 +3129,18 @@ void Plater::priv::show_action_buttons(const bool is_ready_to_slice) const
|
||||||
// when a background processing is ON, export_btn and/or send_btn are showing
|
// when a background processing is ON, export_btn and/or send_btn are showing
|
||||||
if (wxGetApp().app_config->get("background_processing") == "1")
|
if (wxGetApp().app_config->get("background_processing") == "1")
|
||||||
{
|
{
|
||||||
sidebar->show_reslice(false);
|
if (sidebar->show_reslice(false) |
|
||||||
sidebar->show_export(true);
|
sidebar->show_export(true) |
|
||||||
sidebar->show_send(send_gcode_shown);
|
sidebar->show_send(send_gcode_shown))
|
||||||
}
|
sidebar->Layout();
|
||||||
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
sidebar->show_reslice(is_ready_to_slice);
|
if (sidebar->show_reslice(is_ready_to_slice) |
|
||||||
sidebar->show_export(!is_ready_to_slice);
|
sidebar->show_export(!is_ready_to_slice) |
|
||||||
sidebar->show_send(send_gcode_shown && !is_ready_to_slice);
|
sidebar->show_send(send_gcode_shown && !is_ready_to_slice))
|
||||||
}
|
sidebar->Layout();
|
||||||
sidebar->Layout();
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void Sidebar::set_btn_label(const ActionButtonType btn_type, const wxString& label) const
|
void Sidebar::set_btn_label(const ActionButtonType btn_type, const wxString& label) const
|
||||||
|
|
|
@ -96,9 +96,9 @@ public:
|
||||||
void show_sliced_info_sizer(const bool show);
|
void show_sliced_info_sizer(const bool show);
|
||||||
void enable_buttons(bool enable);
|
void enable_buttons(bool enable);
|
||||||
void set_btn_label(const ActionButtonType btn_type, const wxString& label) const;
|
void set_btn_label(const ActionButtonType btn_type, const wxString& label) const;
|
||||||
void show_reslice(bool show) const;
|
bool show_reslice(bool show) const;
|
||||||
void show_export(bool show) const;
|
bool show_export(bool show) const;
|
||||||
void show_send(bool show) const;
|
bool show_send(bool show) const;
|
||||||
bool is_multifilament();
|
bool is_multifilament();
|
||||||
void update_mode();
|
void update_mode();
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue