2019-03-20 12:51:25 +00:00
|
|
|
#include "libslic3r/libslic3r.h"
|
|
|
|
#include "GLGizmosManager.hpp"
|
|
|
|
#include "slic3r/GUI/GLCanvas3D.hpp"
|
|
|
|
#include "slic3r/GUI/3DScene.hpp"
|
2020-03-02 09:58:46 +00:00
|
|
|
#include "slic3r/GUI/Camera.hpp"
|
2019-03-26 11:30:17 +00:00
|
|
|
#include "slic3r/GUI/GUI_App.hpp"
|
|
|
|
#include "slic3r/GUI/GUI_ObjectManipulation.hpp"
|
2020-05-26 15:42:57 +00:00
|
|
|
#include "slic3r/GUI/Plater.hpp"
|
2019-08-22 14:44:49 +00:00
|
|
|
#include "slic3r/Utils/UndoRedo.hpp"
|
2020-04-07 14:19:53 +00:00
|
|
|
|
|
|
|
#include "slic3r/GUI/Gizmos/GLGizmoMove.hpp"
|
|
|
|
#include "slic3r/GUI/Gizmos/GLGizmoScale.hpp"
|
|
|
|
#include "slic3r/GUI/Gizmos/GLGizmoRotate.hpp"
|
|
|
|
#include "slic3r/GUI/Gizmos/GLGizmoFlatten.hpp"
|
|
|
|
#include "slic3r/GUI/Gizmos/GLGizmoSlaSupports.hpp"
|
2020-04-07 17:44:29 +00:00
|
|
|
#include "slic3r/GUI/Gizmos/GLGizmoFdmSupports.hpp"
|
2020-04-07 14:19:53 +00:00
|
|
|
#include "slic3r/GUI/Gizmos/GLGizmoCut.hpp"
|
|
|
|
#include "slic3r/GUI/Gizmos/GLGizmoHollow.hpp"
|
2020-08-31 05:25:12 +00:00
|
|
|
#include "slic3r/GUI/Gizmos/GLGizmoSeam.hpp"
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2020-05-26 09:48:09 +00:00
|
|
|
#include "libslic3r/Model.hpp"
|
2020-06-16 14:58:41 +00:00
|
|
|
#include "libslic3r/PresetBundle.hpp"
|
2020-05-26 09:48:09 +00:00
|
|
|
|
2019-03-26 11:30:17 +00:00
|
|
|
#include <wx/glcanvas.h>
|
2019-03-20 12:51:25 +00:00
|
|
|
|
|
|
|
namespace Slic3r {
|
|
|
|
namespace GUI {
|
|
|
|
|
2019-07-12 08:13:35 +00:00
|
|
|
const float GLGizmosManager::Default_Icons_Size = 64;
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-07-17 10:43:27 +00:00
|
|
|
GLGizmosManager::GLGizmosManager(GLCanvas3D& parent)
|
|
|
|
: m_parent(parent)
|
2019-07-17 06:38:48 +00:00
|
|
|
, m_enabled(false)
|
2019-03-20 12:51:25 +00:00
|
|
|
, m_icons_texture_dirty(true)
|
|
|
|
, m_current(Undefined)
|
2019-03-26 08:01:04 +00:00
|
|
|
, m_tooltip("")
|
2019-07-17 06:38:48 +00:00
|
|
|
, m_serializing(false)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
std::vector<size_t> GLGizmosManager::get_selectable_idxs() const
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-09-02 12:25:48 +00:00
|
|
|
std::vector<size_t> out;
|
|
|
|
for (size_t i=0; i<m_gizmos.size(); ++i)
|
|
|
|
if (m_gizmos[i]->is_selectable())
|
|
|
|
out.push_back(i);
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<size_t> GLGizmosManager::get_activable_idxs() const
|
|
|
|
{
|
|
|
|
std::vector<size_t> out;
|
|
|
|
for (size_t i=0; i<m_gizmos.size(); ++i)
|
|
|
|
if (m_gizmos[i]->is_activable())
|
|
|
|
out.push_back(i);
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t GLGizmosManager::get_gizmo_idx_from_mouse(const Vec2d& mouse_pos) const
|
|
|
|
{
|
|
|
|
if (! m_enabled)
|
|
|
|
return Undefined;
|
|
|
|
|
|
|
|
float cnv_h = (float)m_parent.get_canvas_size().get_height();
|
2019-12-12 10:11:11 +00:00
|
|
|
float height = get_scaled_total_height();
|
|
|
|
float icons_size = m_layout.scaled_icons_size();
|
|
|
|
float border = m_layout.scaled_border();
|
|
|
|
float stride_y = m_layout.scaled_stride_y();
|
|
|
|
float top_y = 0.5f * (cnv_h - height) + border;
|
2019-09-02 12:25:48 +00:00
|
|
|
|
|
|
|
// is mouse horizontally in the area?
|
2019-12-12 10:11:11 +00:00
|
|
|
if ((border <= (float)mouse_pos(0) && ((float)mouse_pos(0) <= border + icons_size))) {
|
2019-09-02 12:25:48 +00:00
|
|
|
// which icon is it on?
|
2019-12-12 10:11:11 +00:00
|
|
|
size_t from_top = (size_t)((float)mouse_pos(1) - top_y) / stride_y;
|
2019-09-02 12:25:48 +00:00
|
|
|
// is it really on the icon or already past the border?
|
2019-12-12 10:11:11 +00:00
|
|
|
if ((float)mouse_pos(1) <= top_y + from_top * stride_y + icons_size) {
|
2019-09-02 12:25:48 +00:00
|
|
|
std::vector<size_t> selectable = get_selectable_idxs();
|
|
|
|
if (from_top < selectable.size())
|
|
|
|
return selectable[from_top];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Undefined;
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 10:43:27 +00:00
|
|
|
bool GLGizmosManager::init()
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
|
|
|
m_background_texture.metadata.filename = "toolbar_background.png";
|
|
|
|
m_background_texture.metadata.left = 16;
|
|
|
|
m_background_texture.metadata.top = 16;
|
|
|
|
m_background_texture.metadata.right = 16;
|
|
|
|
m_background_texture.metadata.bottom = 16;
|
|
|
|
|
|
|
|
if (!m_background_texture.metadata.filename.empty())
|
|
|
|
{
|
2019-07-19 09:18:19 +00:00
|
|
|
if (!m_background_texture.texture.load_from_file(resources_dir() + "/icons/" + m_background_texture.metadata.filename, false, GLTexture::SingleThreaded, false))
|
2019-03-20 12:51:25 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-01-07 13:52:04 +00:00
|
|
|
// Order of gizmos in the vector must match order in EType!
|
2019-09-02 12:25:48 +00:00
|
|
|
m_gizmos.emplace_back(new GLGizmoMove3D(m_parent, "move.svg", 0));
|
|
|
|
m_gizmos.emplace_back(new GLGizmoScale3D(m_parent, "scale.svg", 1));
|
|
|
|
m_gizmos.emplace_back(new GLGizmoRotate3D(m_parent, "rotate.svg", 2));
|
|
|
|
m_gizmos.emplace_back(new GLGizmoFlatten(m_parent, "place.svg", 3));
|
|
|
|
m_gizmos.emplace_back(new GLGizmoCut(m_parent, "cut.svg", 4));
|
2020-02-27 10:26:38 +00:00
|
|
|
m_gizmos.emplace_back(new GLGizmoHollow(m_parent, "hollow.svg", 5));
|
|
|
|
m_gizmos.emplace_back(new GLGizmoSlaSupports(m_parent, "sla_supports.svg", 6));
|
2020-11-02 07:30:33 +00:00
|
|
|
m_gizmos.emplace_back(new GLGizmoFdmSupports(m_parent, "fdm_supports.svg", 7));
|
2020-08-31 05:25:12 +00:00
|
|
|
m_gizmos.emplace_back(new GLGizmoSeam(m_parent, "seam.svg", 8));
|
2020-02-27 10:26:38 +00:00
|
|
|
|
2020-04-02 15:07:15 +00:00
|
|
|
m_common_gizmos_data.reset(new CommonGizmosDataPool(&m_parent));
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
for (auto& gizmo : m_gizmos) {
|
|
|
|
if (! gizmo->init()) {
|
|
|
|
m_gizmos.clear();
|
|
|
|
return false;
|
|
|
|
}
|
2020-04-02 15:07:15 +00:00
|
|
|
gizmo->set_common_data_pool(m_common_gizmos_data.get());
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-08-19 09:36:55 +00:00
|
|
|
m_current = Undefined;
|
|
|
|
m_hover = Undefined;
|
2019-03-20 12:51:25 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void GLGizmosManager::set_overlay_icon_size(float size)
|
|
|
|
{
|
2019-12-12 10:11:11 +00:00
|
|
|
if (m_layout.icons_size != size)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-12-12 10:11:11 +00:00
|
|
|
m_layout.icons_size = size;
|
2019-03-20 12:51:25 +00:00
|
|
|
m_icons_texture_dirty = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void GLGizmosManager::set_overlay_scale(float scale)
|
|
|
|
{
|
2019-12-12 10:11:11 +00:00
|
|
|
if (m_layout.scale != scale)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-12-12 10:11:11 +00:00
|
|
|
m_layout.scale = scale;
|
2019-03-20 12:51:25 +00:00
|
|
|
m_icons_texture_dirty = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
void GLGizmosManager::refresh_on_off_state()
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (m_serializing || m_current == Undefined || m_gizmos.empty())
|
2019-07-17 06:38:48 +00:00
|
|
|
return;
|
|
|
|
|
2020-08-05 11:30:05 +00:00
|
|
|
if (m_current != Undefined
|
|
|
|
&& (! m_gizmos[m_current]->is_activable() || ! m_gizmos[m_current]->is_selectable())) {
|
2019-09-02 12:25:48 +00:00
|
|
|
activate_gizmo(Undefined);
|
2020-08-05 11:30:05 +00:00
|
|
|
update_data();
|
|
|
|
}
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GLGizmosManager::reset_all_states()
|
|
|
|
{
|
2019-09-02 12:25:48 +00:00
|
|
|
if (! m_enabled || m_serializing)
|
2019-07-17 06:38:48 +00:00
|
|
|
return;
|
|
|
|
|
2019-08-19 10:11:26 +00:00
|
|
|
activate_gizmo(Undefined);
|
2019-08-19 09:36:55 +00:00
|
|
|
m_hover = Undefined;
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GLGizmosManager::set_hover_id(int id)
|
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (!m_enabled || m_current == Undefined)
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
2019-08-19 09:36:55 +00:00
|
|
|
m_gizmos[m_current]->set_hover_id(id);
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GLGizmosManager::enable_grabber(EType type, unsigned int id, bool enable)
|
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (!m_enabled || type == Undefined || m_gizmos.empty())
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
2019-08-19 09:36:55 +00:00
|
|
|
if (enable)
|
|
|
|
m_gizmos[type]->enable_grabber(id);
|
|
|
|
else
|
|
|
|
m_gizmos[type]->disable_grabber(id);
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 10:06:23 +00:00
|
|
|
void GLGizmosManager::update(const Linef3& mouse_ray, const Point& mouse_pos)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-07-17 10:43:27 +00:00
|
|
|
if (!m_enabled)
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
GLGizmoBase* curr = get_current();
|
|
|
|
if (curr != nullptr)
|
2019-07-17 10:06:23 +00:00
|
|
|
curr->update(GLGizmoBase::UpdateData(mouse_ray, mouse_pos));
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
void GLGizmosManager::update_data()
|
2019-03-26 14:55:47 +00:00
|
|
|
{
|
2019-07-17 10:43:27 +00:00
|
|
|
if (!m_enabled)
|
2019-03-26 14:55:47 +00:00
|
|
|
return;
|
|
|
|
|
2019-07-17 10:43:27 +00:00
|
|
|
const Selection& selection = m_parent.get_selection();
|
2019-03-26 14:55:47 +00:00
|
|
|
|
2019-04-26 13:34:26 +00:00
|
|
|
bool is_wipe_tower = selection.is_wipe_tower();
|
|
|
|
enable_grabber(Move, 2, !is_wipe_tower);
|
|
|
|
enable_grabber(Rotate, 0, !is_wipe_tower);
|
|
|
|
enable_grabber(Rotate, 1, !is_wipe_tower);
|
2019-04-29 12:32:02 +00:00
|
|
|
|
2019-03-26 14:55:47 +00:00
|
|
|
bool enable_scale_xyz = selection.is_single_full_instance() || selection.is_single_volume() || selection.is_single_modifier();
|
2019-07-23 09:22:54 +00:00
|
|
|
for (unsigned int i = 0; i < 6; ++i)
|
2019-03-26 14:55:47 +00:00
|
|
|
{
|
|
|
|
enable_grabber(Scale, i, enable_scale_xyz);
|
|
|
|
}
|
|
|
|
|
2020-08-05 11:30:05 +00:00
|
|
|
if (m_common_gizmos_data)
|
|
|
|
m_common_gizmos_data->update(get_current()
|
|
|
|
? get_current()->get_requirements()
|
|
|
|
: CommonGizmosDataID(0));
|
2020-04-03 11:27:04 +00:00
|
|
|
|
2019-03-26 14:55:47 +00:00
|
|
|
if (selection.is_single_full_instance())
|
|
|
|
{
|
|
|
|
// all volumes in the selection belongs to the same instance, any of them contains the needed data, so we take the first
|
|
|
|
const GLVolume* volume = selection.get_volume(*selection.get_volume_idxs().begin());
|
|
|
|
set_scale(volume->get_instance_scaling_factor());
|
|
|
|
set_rotation(Vec3d::Zero());
|
2019-03-28 08:30:01 +00:00
|
|
|
ModelObject* model_object = selection.get_model()->objects[selection.get_object_idx()];
|
2019-03-26 14:55:47 +00:00
|
|
|
set_flattening_data(model_object);
|
2019-07-17 06:38:48 +00:00
|
|
|
set_sla_support_data(model_object);
|
2020-08-26 09:33:41 +00:00
|
|
|
set_painter_gizmo_data();
|
2019-03-26 14:55:47 +00:00
|
|
|
}
|
|
|
|
else if (selection.is_single_volume() || selection.is_single_modifier())
|
|
|
|
{
|
|
|
|
const GLVolume* volume = selection.get_volume(*selection.get_volume_idxs().begin());
|
|
|
|
set_scale(volume->get_volume_scaling_factor());
|
|
|
|
set_rotation(Vec3d::Zero());
|
|
|
|
set_flattening_data(nullptr);
|
2019-07-17 06:38:48 +00:00
|
|
|
set_sla_support_data(nullptr);
|
2020-08-26 09:33:41 +00:00
|
|
|
set_painter_gizmo_data();
|
2019-03-26 14:55:47 +00:00
|
|
|
}
|
2019-04-26 13:34:26 +00:00
|
|
|
else if (is_wipe_tower)
|
|
|
|
{
|
|
|
|
DynamicPrintConfig& config = wxGetApp().preset_bundle->prints.get_edited_preset().config;
|
|
|
|
set_scale(Vec3d::Ones());
|
|
|
|
set_rotation(Vec3d(0., 0., (M_PI/180.) * dynamic_cast<const ConfigOptionFloat*>(config.option("wipe_tower_rotation_angle"))->value));
|
|
|
|
set_flattening_data(nullptr);
|
2019-07-17 06:38:48 +00:00
|
|
|
set_sla_support_data(nullptr);
|
2020-08-26 09:33:41 +00:00
|
|
|
set_painter_gizmo_data();
|
2019-04-26 13:34:26 +00:00
|
|
|
}
|
2019-03-26 14:55:47 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
set_scale(Vec3d::Ones());
|
|
|
|
set_rotation(Vec3d::Zero());
|
2019-03-28 08:30:01 +00:00
|
|
|
set_flattening_data(selection.is_from_single_object() ? selection.get_model()->objects[selection.get_object_idx()] : nullptr);
|
2020-03-13 14:58:40 +00:00
|
|
|
set_sla_support_data(selection.is_from_single_instance() ? selection.get_model()->objects[selection.get_object_idx()] : nullptr);
|
2020-08-26 09:33:41 +00:00
|
|
|
set_painter_gizmo_data();
|
2019-03-26 14:55:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-20 12:51:25 +00:00
|
|
|
bool GLGizmosManager::is_running() const
|
|
|
|
{
|
|
|
|
if (!m_enabled)
|
|
|
|
return false;
|
|
|
|
|
2019-08-19 10:11:26 +00:00
|
|
|
//GLGizmoBase* curr = get_current();
|
|
|
|
//return (curr != nullptr) ? (curr->get_state() == GLGizmoBase::On) : false;
|
|
|
|
return m_current != Undefined;
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
bool GLGizmosManager::handle_shortcut(int key)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-07-17 10:43:27 +00:00
|
|
|
if (!m_enabled)
|
2019-07-17 06:38:48 +00:00
|
|
|
return false;
|
|
|
|
|
2019-07-17 10:43:27 +00:00
|
|
|
if (m_parent.get_selection().is_empty())
|
2019-03-20 12:51:25 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
bool handled = false;
|
2019-08-22 23:08:14 +00:00
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
for (size_t idx : get_selectable_idxs()) {
|
|
|
|
int it_key = m_gizmos[idx]->get_shortcut_key();
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
if (m_gizmos[idx]->is_activable() && ((it_key == key - 64) || (it_key == key - 96))) {
|
|
|
|
activate_gizmo(m_current == idx ? Undefined : (EType)idx);
|
2019-03-20 12:51:25 +00:00
|
|
|
handled = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return handled;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GLGizmosManager::is_dragging() const
|
|
|
|
{
|
2019-09-02 12:25:48 +00:00
|
|
|
if (! m_enabled || m_current == Undefined)
|
2019-03-20 12:51:25 +00:00
|
|
|
return false;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
return m_gizmos[m_current]->is_dragging();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
void GLGizmosManager::start_dragging()
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-09-02 12:25:48 +00:00
|
|
|
if (! m_enabled || m_current == Undefined)
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
2019-09-02 12:25:48 +00:00
|
|
|
m_gizmos[m_current]->start_dragging();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GLGizmosManager::stop_dragging()
|
|
|
|
{
|
2019-09-02 12:25:48 +00:00
|
|
|
if (! m_enabled || m_current == Undefined)
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
m_gizmos[m_current]->stop_dragging();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Vec3d GLGizmosManager::get_displacement() const
|
|
|
|
{
|
|
|
|
if (!m_enabled)
|
|
|
|
return Vec3d::Zero();
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
return dynamic_cast<GLGizmoMove3D*>(m_gizmos[Move].get())->get_displacement();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Vec3d GLGizmosManager::get_scale() const
|
|
|
|
{
|
|
|
|
if (!m_enabled)
|
|
|
|
return Vec3d::Ones();
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
return dynamic_cast<GLGizmoScale3D*>(m_gizmos[Scale].get())->get_scale();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GLGizmosManager::set_scale(const Vec3d& scale)
|
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (!m_enabled || m_gizmos.empty())
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
dynamic_cast<GLGizmoScale3D*>(m_gizmos[Scale].get())->set_scale(scale);
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-05-10 09:13:55 +00:00
|
|
|
Vec3d GLGizmosManager::get_scale_offset() const
|
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (!m_enabled || m_gizmos.empty())
|
2019-05-10 11:07:59 +00:00
|
|
|
return Vec3d::Zero();
|
2019-05-10 09:13:55 +00:00
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
return dynamic_cast<GLGizmoScale3D*>(m_gizmos[Scale].get())->get_offset();
|
2019-05-10 09:13:55 +00:00
|
|
|
}
|
|
|
|
|
2019-03-20 12:51:25 +00:00
|
|
|
Vec3d GLGizmosManager::get_rotation() const
|
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (!m_enabled || m_gizmos.empty())
|
2019-03-20 12:51:25 +00:00
|
|
|
return Vec3d::Zero();
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
return dynamic_cast<GLGizmoRotate3D*>(m_gizmos[Rotate].get())->get_rotation();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GLGizmosManager::set_rotation(const Vec3d& rotation)
|
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (!m_enabled || m_gizmos.empty())
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
2019-09-02 12:25:48 +00:00
|
|
|
dynamic_cast<GLGizmoRotate3D*>(m_gizmos[Rotate].get())->set_rotation(rotation);
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Vec3d GLGizmosManager::get_flattening_normal() const
|
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (!m_enabled || m_gizmos.empty())
|
2019-03-20 12:51:25 +00:00
|
|
|
return Vec3d::Zero();
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
return dynamic_cast<GLGizmoFlatten*>(m_gizmos[Flatten].get())->get_flattening_normal();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GLGizmosManager::set_flattening_data(const ModelObject* model_object)
|
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (!m_enabled || m_gizmos.empty())
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
dynamic_cast<GLGizmoFlatten*>(m_gizmos[Flatten].get())->set_flattening_data(model_object);
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
void GLGizmosManager::set_sla_support_data(ModelObject* model_object)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2020-02-27 09:21:36 +00:00
|
|
|
if (! m_enabled
|
|
|
|
|| m_gizmos.empty()
|
|
|
|
|| wxGetApp().preset_bundle->printers.get_edited_preset().printer_technology() != ptSLA)
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
2020-04-07 07:29:34 +00:00
|
|
|
auto* gizmo_hollow = dynamic_cast<GLGizmoHollow*>(m_gizmos[Hollow].get());
|
2020-04-07 13:38:24 +00:00
|
|
|
auto* gizmo_supports = dynamic_cast<GLGizmoSlaSupports*>(m_gizmos[SlaSupports].get());
|
2020-02-06 13:46:02 +00:00
|
|
|
gizmo_hollow->set_sla_support_data(model_object, m_parent.get_selection());
|
2020-04-07 13:38:24 +00:00
|
|
|
gizmo_supports->set_sla_support_data(model_object, m_parent.get_selection());
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2020-08-26 09:33:41 +00:00
|
|
|
void GLGizmosManager::set_painter_gizmo_data()
|
2019-09-20 10:42:08 +00:00
|
|
|
{
|
|
|
|
if (!m_enabled || m_gizmos.empty())
|
|
|
|
return;
|
|
|
|
|
2020-08-26 09:33:41 +00:00
|
|
|
dynamic_cast<GLGizmoFdmSupports*>(m_gizmos[FdmSupports].get())->set_painter_gizmo_data(m_parent.get_selection());
|
2020-08-31 05:25:12 +00:00
|
|
|
dynamic_cast<GLGizmoSeam*>(m_gizmos[Seam].get())->set_painter_gizmo_data(m_parent.get_selection());
|
2019-09-20 10:42:08 +00:00
|
|
|
}
|
|
|
|
|
2019-03-20 12:51:25 +00:00
|
|
|
// Returns true if the gizmo used the event to do something, false otherwise.
|
2019-04-08 13:34:42 +00:00
|
|
|
bool GLGizmosManager::gizmo_event(SLAGizmoEventType action, const Vec2d& mouse_position, bool shift_down, bool alt_down, bool control_down)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (!m_enabled || m_gizmos.empty())
|
2019-03-20 12:51:25 +00:00
|
|
|
return false;
|
|
|
|
|
2019-11-04 16:01:26 +00:00
|
|
|
if (m_current == SlaSupports)
|
|
|
|
return dynamic_cast<GLGizmoSlaSupports*>(m_gizmos[SlaSupports].get())->gizmo_event(action, mouse_position, shift_down, alt_down, control_down);
|
2019-09-26 11:30:22 +00:00
|
|
|
else if (m_current == Hollow)
|
2019-11-04 16:01:26 +00:00
|
|
|
return dynamic_cast<GLGizmoHollow*>(m_gizmos[Hollow].get())->gizmo_event(action, mouse_position, shift_down, alt_down, control_down);
|
2019-09-26 11:30:22 +00:00
|
|
|
else if (m_current == FdmSupports)
|
|
|
|
return dynamic_cast<GLGizmoFdmSupports*>(m_gizmos[FdmSupports].get())->gizmo_event(action, mouse_position, shift_down, alt_down, control_down);
|
2020-08-31 05:25:12 +00:00
|
|
|
else if (m_current == Seam)
|
|
|
|
return dynamic_cast<GLGizmoSeam*>(m_gizmos[Seam].get())->gizmo_event(action, mouse_position, shift_down, alt_down, control_down);
|
2019-09-26 11:30:22 +00:00
|
|
|
else
|
|
|
|
return false;
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-09-20 10:42:08 +00:00
|
|
|
ClippingPlane GLGizmosManager::get_clipping_plane() const
|
2019-04-08 08:50:10 +00:00
|
|
|
{
|
2020-04-08 15:22:27 +00:00
|
|
|
if (! m_common_gizmos_data
|
|
|
|
|| ! m_common_gizmos_data->object_clipper()
|
2020-04-07 22:34:05 +00:00
|
|
|
|| m_common_gizmos_data->object_clipper()->get_position() == 0.)
|
2019-04-08 08:50:10 +00:00
|
|
|
return ClippingPlane::ClipsNothing();
|
2020-04-07 13:55:54 +00:00
|
|
|
else {
|
|
|
|
const ClippingPlane& clp = *m_common_gizmos_data->object_clipper()->get_clipping_plane();
|
|
|
|
return ClippingPlane(-clp.get_normal(), clp.get_data()[3]);
|
|
|
|
}
|
2019-04-08 08:50:10 +00:00
|
|
|
}
|
|
|
|
|
2019-08-22 14:44:49 +00:00
|
|
|
bool GLGizmosManager::wants_reslice_supports_on_undo() const
|
|
|
|
{
|
|
|
|
return (m_current == SlaSupports
|
2019-09-02 12:25:48 +00:00
|
|
|
&& dynamic_cast<const GLGizmoSlaSupports*>(m_gizmos.at(SlaSupports).get())->has_backend_supports());
|
2019-08-22 14:44:49 +00:00
|
|
|
}
|
2019-04-08 08:50:10 +00:00
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
void GLGizmosManager::render_current_gizmo() const
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-09-02 12:25:48 +00:00
|
|
|
if (!m_enabled || m_current == Undefined)
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
m_gizmos[m_current]->render();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2020-10-05 14:27:16 +00:00
|
|
|
void GLGizmosManager::render_painter_gizmo() const
|
|
|
|
{
|
|
|
|
// This function shall only be called when current gizmo is
|
|
|
|
// derived from GLGizmoPainterBase.
|
|
|
|
|
|
|
|
if (!m_enabled || m_current == Undefined)
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto* gizmo = dynamic_cast<GLGizmoPainterBase*>(get_current());
|
|
|
|
assert(gizmo); // check the precondition
|
|
|
|
gizmo->render_painter_gizmo();
|
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
void GLGizmosManager::render_current_gizmo_for_picking_pass() const
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-09-02 12:25:48 +00:00
|
|
|
if (! m_enabled || m_current == Undefined)
|
2019-09-20 10:42:08 +00:00
|
|
|
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
m_gizmos[m_current]->render_for_picking();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
void GLGizmosManager::render_overlay() const
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-07-17 10:43:27 +00:00
|
|
|
if (!m_enabled)
|
2019-03-20 12:51:25 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (m_icons_texture_dirty)
|
|
|
|
generate_icons_texture();
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
do_render_overlay();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2020-03-16 11:45:39 +00:00
|
|
|
std::string GLGizmosManager::get_tooltip() const
|
|
|
|
{
|
|
|
|
if (!m_tooltip.empty())
|
|
|
|
return m_tooltip;
|
|
|
|
|
|
|
|
const GLGizmoBase* curr = get_current();
|
|
|
|
return (curr != nullptr) ? curr->get_tooltip() : "";
|
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
bool GLGizmosManager::on_mouse_wheel(wxMouseEvent& evt)
|
2019-04-17 08:16:39 +00:00
|
|
|
{
|
|
|
|
bool processed = false;
|
|
|
|
|
2020-08-31 05:25:12 +00:00
|
|
|
if (m_current == SlaSupports || m_current == Hollow || m_current == FdmSupports || m_current == Seam) {
|
2019-04-17 08:16:39 +00:00
|
|
|
float rot = (float)evt.GetWheelRotation() / (float)evt.GetWheelDelta();
|
|
|
|
if (gizmo_event((rot > 0.f ? SLAGizmoEventType::MouseWheelUp : SLAGizmoEventType::MouseWheelDown), Vec2d::Zero(), evt.ShiftDown(), evt.AltDown(), evt.ControlDown()))
|
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return processed;
|
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
bool GLGizmosManager::on_mouse(wxMouseEvent& evt)
|
2019-03-26 08:01:04 +00:00
|
|
|
{
|
2019-10-10 14:00:32 +00:00
|
|
|
// used to set a right up event as processed when needed
|
|
|
|
static bool pending_right_up = false;
|
|
|
|
|
2019-03-26 11:30:17 +00:00
|
|
|
Point pos(evt.GetX(), evt.GetY());
|
2019-03-26 08:01:04 +00:00
|
|
|
Vec2d mouse_pos((double)evt.GetX(), (double)evt.GetY());
|
|
|
|
|
2019-07-17 10:43:27 +00:00
|
|
|
Selection& selection = m_parent.get_selection();
|
2019-03-26 11:30:17 +00:00
|
|
|
int selected_object_idx = selection.get_object_idx();
|
2019-03-26 08:01:04 +00:00
|
|
|
bool processed = false;
|
|
|
|
|
2020-10-19 13:55:16 +00:00
|
|
|
// when control is down we allow scene pan and rotation even when clicking over some object
|
|
|
|
bool control_down = evt.CmdDown();
|
|
|
|
|
2019-03-26 08:01:04 +00:00
|
|
|
// mouse anywhere
|
|
|
|
if (evt.Moving())
|
2019-07-17 06:38:48 +00:00
|
|
|
m_tooltip = update_hover_state(mouse_pos);
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.LeftUp()) {
|
|
|
|
if (m_mouse_capture.left) {
|
2020-03-19 13:05:49 +00:00
|
|
|
processed = true;
|
|
|
|
m_mouse_capture.left = false;
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (is_dragging()) {
|
2020-03-19 13:05:49 +00:00
|
|
|
switch (m_current) {
|
2020-10-21 09:25:28 +00:00
|
|
|
case Move: { m_parent.do_move(L("Gizmo-Move")); break; }
|
|
|
|
case Scale: { m_parent.do_scale(L("Gizmo-Scale")); break; }
|
|
|
|
case Rotate: { m_parent.do_rotate(L("Gizmo-Rotate")); break; }
|
2020-03-19 13:05:49 +00:00
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
|
|
|
|
stop_dragging();
|
|
|
|
update_data();
|
|
|
|
|
|
|
|
wxGetApp().obj_manipul()->set_dirty();
|
|
|
|
// Let the plater know that the dragging finished, so a delayed refresh
|
|
|
|
// of the scene with the background processing data should be performed.
|
|
|
|
m_parent.post_event(SimpleEvent(EVT_GLCANVAS_MOUSE_DRAGGING_FINISHED));
|
|
|
|
// updates camera target constraints
|
|
|
|
m_parent.refresh_camera_scene_box();
|
|
|
|
|
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
// else
|
|
|
|
// return false;
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.MiddleUp()) {
|
|
|
|
if (m_mouse_capture.middle) {
|
2020-03-19 13:05:49 +00:00
|
|
|
processed = true;
|
|
|
|
m_mouse_capture.middle = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.RightUp()) {
|
|
|
|
if (pending_right_up) {
|
2020-03-19 13:05:49 +00:00
|
|
|
pending_right_up = false;
|
|
|
|
return true;
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
if (m_mouse_capture.right) {
|
2020-03-19 13:05:49 +00:00
|
|
|
processed = true;
|
|
|
|
m_mouse_capture.right = false;
|
|
|
|
}
|
2020-04-08 12:59:53 +00:00
|
|
|
// else
|
|
|
|
// return false;
|
2020-03-19 13:05:49 +00:00
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.Dragging() && !is_dragging()) {
|
2020-03-19 13:05:49 +00:00
|
|
|
if (m_mouse_capture.any())
|
|
|
|
// if the button down was done on this toolbar, prevent from dragging into the scene
|
|
|
|
processed = true;
|
|
|
|
// else
|
|
|
|
// return false;
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.Dragging() && is_dragging()) {
|
2020-03-19 13:05:49 +00:00
|
|
|
if (!m_parent.get_wxglcanvas()->HasCapture())
|
|
|
|
m_parent.get_wxglcanvas()->CaptureMouse();
|
|
|
|
|
|
|
|
m_parent.set_mouse_as_dragging();
|
|
|
|
update(m_parent.mouse_ray(pos), pos);
|
|
|
|
|
|
|
|
switch (m_current)
|
|
|
|
{
|
|
|
|
case Move:
|
|
|
|
{
|
|
|
|
// Apply new temporary offset
|
|
|
|
selection.translate(get_displacement());
|
|
|
|
wxGetApp().obj_manipul()->set_dirty();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Scale:
|
|
|
|
{
|
|
|
|
// Apply new temporary scale factors
|
|
|
|
TransformationType transformation_type(TransformationType::Local_Absolute_Joint);
|
|
|
|
if (evt.AltDown())
|
|
|
|
transformation_type.set_independent();
|
|
|
|
selection.scale(get_scale(), transformation_type);
|
2020-10-19 13:55:16 +00:00
|
|
|
if (control_down)
|
2020-03-19 13:05:49 +00:00
|
|
|
selection.translate(get_scale_offset(), true);
|
|
|
|
wxGetApp().obj_manipul()->set_dirty();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Rotate:
|
|
|
|
{
|
|
|
|
// Apply new temporary rotations
|
|
|
|
TransformationType transformation_type(TransformationType::World_Relative_Joint);
|
|
|
|
if (evt.AltDown())
|
|
|
|
transformation_type.set_independent();
|
|
|
|
selection.rotate(get_rotation(), transformation_type);
|
|
|
|
wxGetApp().obj_manipul()->set_dirty();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_parent.set_as_dirty();
|
|
|
|
processed = true;
|
|
|
|
}
|
2019-03-26 08:01:04 +00:00
|
|
|
|
2020-10-19 13:55:16 +00:00
|
|
|
if (get_gizmo_idx_from_mouse(mouse_pos) == Undefined) {
|
2019-03-26 08:01:04 +00:00
|
|
|
// mouse is outside the toolbar
|
|
|
|
m_tooltip = "";
|
2019-03-26 11:30:17 +00:00
|
|
|
|
2020-10-19 13:55:16 +00:00
|
|
|
if (evt.LeftDown() && (!control_down || grabber_contains_mouse())) {
|
|
|
|
if ((m_current == SlaSupports || m_current == Hollow || m_current == FdmSupports || m_current == Seam)
|
|
|
|
&& gizmo_event(SLAGizmoEventType::LeftDown, mouse_pos, evt.ShiftDown(), evt.AltDown()))
|
2019-03-26 11:30:17 +00:00
|
|
|
// the gizmo got the event and took some action, there is no need to do anything more
|
|
|
|
processed = true;
|
2019-09-02 12:25:48 +00:00
|
|
|
else if (!selection.is_empty() && grabber_contains_mouse()) {
|
2019-07-17 06:38:48 +00:00
|
|
|
update_data();
|
2019-03-26 11:30:17 +00:00
|
|
|
selection.start_dragging();
|
2019-07-17 06:38:48 +00:00
|
|
|
start_dragging();
|
2019-03-26 11:30:17 +00:00
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
if (m_current == Flatten) {
|
2019-03-26 11:30:17 +00:00
|
|
|
// Rotate the object so the normal points downward:
|
2019-07-17 12:13:50 +00:00
|
|
|
m_parent.do_flatten(get_flattening_normal(), L("Gizmo-Place on Face"));
|
2019-05-03 10:36:26 +00:00
|
|
|
wxGetApp().obj_manipul()->set_dirty();
|
2019-03-26 11:30:17 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 10:43:27 +00:00
|
|
|
m_parent.set_as_dirty();
|
2019-03-26 11:30:17 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.RightDown() && selected_object_idx != -1 && (m_current == SlaSupports || m_current == Hollow)
|
|
|
|
&& gizmo_event(SLAGizmoEventType::RightDown, mouse_pos)) {
|
2019-10-10 14:00:32 +00:00
|
|
|
// we need to set the following right up as processed to avoid showing the context menu if the user release the mouse over the object
|
|
|
|
pending_right_up = true;
|
2019-03-26 11:30:17 +00:00
|
|
|
// event was taken care of by the SlaSupports gizmo
|
|
|
|
processed = true;
|
2019-10-10 14:00:32 +00:00
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.RightDown() && !control_down && selected_object_idx != -1 && (m_current == FdmSupports || m_current == Seam)
|
|
|
|
&& gizmo_event(SLAGizmoEventType::RightDown, mouse_pos)) {
|
2020-08-31 05:25:12 +00:00
|
|
|
// event was taken care of by the FdmSupports / Seam gizmo
|
2020-04-08 12:59:53 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
2020-10-21 09:25:28 +00:00
|
|
|
else if (evt.Dragging() && m_parent.get_move_volume_id() != -1
|
2020-08-31 05:25:12 +00:00
|
|
|
&& (m_current == SlaSupports || m_current == Hollow || m_current == FdmSupports || m_current == Seam))
|
2020-02-03 13:39:16 +00:00
|
|
|
// don't allow dragging objects with the Sla gizmo on
|
2019-03-26 11:30:17 +00:00
|
|
|
processed = true;
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.Dragging() && !control_down && (m_current == SlaSupports || m_current == Hollow || m_current == FdmSupports || m_current == Seam)
|
|
|
|
&& gizmo_event(SLAGizmoEventType::Dragging, mouse_pos, evt.ShiftDown(), evt.AltDown())) {
|
2019-03-26 11:30:17 +00:00
|
|
|
// the gizmo got the event and took some action, no need to do anything more here
|
2019-07-17 10:43:27 +00:00
|
|
|
m_parent.set_as_dirty();
|
2019-03-26 11:30:17 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
2020-10-21 09:25:28 +00:00
|
|
|
else if (evt.Dragging() && control_down && (evt.LeftIsDown() || evt.RightIsDown())) {
|
|
|
|
// CTRL has been pressed while already dragging -> stop current action
|
|
|
|
if (evt.LeftIsDown())
|
|
|
|
gizmo_event(SLAGizmoEventType::LeftUp, mouse_pos, evt.ShiftDown(), evt.AltDown(), true);
|
|
|
|
else if (evt.RightIsDown())
|
|
|
|
gizmo_event(SLAGizmoEventType::RightUp, mouse_pos, evt.ShiftDown(), evt.AltDown(), true);
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.LeftUp() && (m_current == SlaSupports || m_current == Hollow || m_current == FdmSupports || m_current == Seam) && !m_parent.is_mouse_dragging()) {
|
2019-09-26 11:30:22 +00:00
|
|
|
// in case SLA/FDM gizmo is selected, we just pass the LeftUp event and stop processing - neither
|
2019-03-26 11:30:17 +00:00
|
|
|
// object moving or selecting is suppressed in that case
|
2020-10-19 13:55:16 +00:00
|
|
|
gizmo_event(SLAGizmoEventType::LeftUp, mouse_pos, evt.ShiftDown(), evt.AltDown(), control_down);
|
2019-03-26 11:30:17 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.LeftUp() && m_current == Flatten && m_gizmos[m_current]->get_hover_id() != -1) {
|
2019-09-09 12:56:55 +00:00
|
|
|
// to avoid to loose the selection when user clicks an the white faces of a different object while the Flatten gizmo is active
|
2019-03-27 11:51:40 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.RightUp() && (m_current == FdmSupports || m_current == Seam) && !m_parent.is_mouse_dragging()) {
|
|
|
|
gizmo_event(SLAGizmoEventType::RightUp, mouse_pos, evt.ShiftDown(), evt.AltDown(), control_down);
|
2020-04-08 12:59:53 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
2019-03-26 08:01:04 +00:00
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else {
|
2019-03-26 08:01:04 +00:00
|
|
|
// mouse inside toolbar
|
2020-10-19 13:55:16 +00:00
|
|
|
if (evt.LeftDown() || evt.LeftDClick()) {
|
2019-03-26 08:01:04 +00:00
|
|
|
m_mouse_capture.left = true;
|
2019-07-17 10:43:27 +00:00
|
|
|
m_mouse_capture.parent = &m_parent;
|
2019-03-26 08:01:04 +00:00
|
|
|
processed = true;
|
2020-10-19 13:55:16 +00:00
|
|
|
if (!selection.is_empty()) {
|
2019-07-17 06:38:48 +00:00
|
|
|
update_on_off_state(mouse_pos);
|
|
|
|
update_data();
|
2019-07-17 10:43:27 +00:00
|
|
|
m_parent.set_as_dirty();
|
2019-03-26 08:01:04 +00:00
|
|
|
}
|
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.MiddleDown()) {
|
2019-03-26 08:01:04 +00:00
|
|
|
m_mouse_capture.middle = true;
|
2019-07-17 10:43:27 +00:00
|
|
|
m_mouse_capture.parent = &m_parent;
|
2019-03-26 08:01:04 +00:00
|
|
|
}
|
2020-10-19 13:55:16 +00:00
|
|
|
else if (evt.RightDown()) {
|
2019-03-26 08:01:04 +00:00
|
|
|
m_mouse_capture.right = true;
|
2019-07-17 10:43:27 +00:00
|
|
|
m_mouse_capture.parent = &m_parent;
|
2019-03-26 08:01:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return processed;
|
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
bool GLGizmosManager::on_char(wxKeyEvent& evt)
|
2019-03-26 13:08:02 +00:00
|
|
|
{
|
|
|
|
// see include/wx/defs.h enum wxKeyCode
|
|
|
|
int keyCode = evt.GetKeyCode();
|
|
|
|
int ctrlMask = wxMOD_CONTROL;
|
|
|
|
|
|
|
|
bool processed = false;
|
|
|
|
|
|
|
|
if ((evt.GetModifiers() & ctrlMask) != 0)
|
|
|
|
{
|
|
|
|
switch (keyCode)
|
|
|
|
{
|
2019-04-10 08:16:04 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
case 'a':
|
|
|
|
case 'A':
|
|
|
|
#else /* __APPLE__ */
|
2019-03-26 13:08:02 +00:00
|
|
|
case WXK_CONTROL_A:
|
2019-04-10 08:16:04 +00:00
|
|
|
#endif /* __APPLE__ */
|
2019-03-26 13:08:02 +00:00
|
|
|
{
|
|
|
|
// Sla gizmo selects all support points
|
2019-11-29 13:09:53 +00:00
|
|
|
if ((m_current == SlaSupports || m_current == Hollow) && gizmo_event(SLAGizmoEventType::SelectAll))
|
2019-03-26 13:08:02 +00:00
|
|
|
processed = true;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!evt.HasModifiers())
|
|
|
|
{
|
|
|
|
switch (keyCode)
|
|
|
|
{
|
|
|
|
// key ESC
|
|
|
|
case WXK_ESCAPE:
|
|
|
|
{
|
2019-05-14 09:57:39 +00:00
|
|
|
if (m_current != Undefined)
|
|
|
|
{
|
|
|
|
if ((m_current != SlaSupports) || !gizmo_event(SLAGizmoEventType::DiscardChanges))
|
|
|
|
reset_all_states();
|
2019-03-26 13:08:02 +00:00
|
|
|
|
2019-05-14 09:57:39 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
2019-03-26 13:08:02 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case WXK_RETURN:
|
|
|
|
{
|
|
|
|
if ((m_current == SlaSupports) && gizmo_event(SLAGizmoEventType::ApplyChanges))
|
|
|
|
processed = true;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2019-04-17 10:24:46 +00:00
|
|
|
|
|
|
|
case 'r' :
|
|
|
|
case 'R' :
|
|
|
|
{
|
2020-08-31 05:25:12 +00:00
|
|
|
if ((m_current == SlaSupports || m_current == Hollow || m_current == FdmSupports || m_current == Seam) && gizmo_event(SLAGizmoEventType::ResetClippingPlane))
|
2019-04-17 10:24:46 +00:00
|
|
|
processed = true;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-03-26 13:08:02 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
case WXK_BACK: // the low cost Apple solutions are not equipped with a Delete key, use Backspace instead.
|
|
|
|
#else /* __APPLE__ */
|
|
|
|
case WXK_DELETE:
|
|
|
|
#endif /* __APPLE__ */
|
|
|
|
{
|
2019-11-05 13:40:22 +00:00
|
|
|
if ((m_current == SlaSupports || m_current == Hollow) && gizmo_event(SLAGizmoEventType::Delete))
|
2019-03-26 13:08:02 +00:00
|
|
|
processed = true;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'A':
|
|
|
|
case 'a':
|
|
|
|
{
|
|
|
|
if (m_current == SlaSupports)
|
|
|
|
{
|
|
|
|
gizmo_event(SLAGizmoEventType::AutomaticGeneration);
|
|
|
|
// set as processed no matter what's returned by gizmo_event() to avoid the calling canvas to process 'A' as arrange
|
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 'M':
|
|
|
|
case 'm':
|
|
|
|
{
|
|
|
|
if ((m_current == SlaSupports) && gizmo_event(SLAGizmoEventType::ManualEditing))
|
|
|
|
processed = true;
|
2020-01-07 13:52:04 +00:00
|
|
|
|
2019-03-26 13:08:02 +00:00
|
|
|
break;
|
|
|
|
}
|
2019-05-22 12:42:38 +00:00
|
|
|
case 'F':
|
|
|
|
case 'f':
|
|
|
|
{
|
|
|
|
if (m_current == Scale)
|
|
|
|
{
|
2019-05-23 07:20:11 +00:00
|
|
|
if (!is_dragging())
|
|
|
|
wxGetApp().plater()->scale_selection_to_fit_print_volume();
|
|
|
|
|
2019-05-22 12:42:38 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2019-03-26 13:08:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-17 07:44:46 +00:00
|
|
|
if (!processed && !evt.HasModifiers())
|
2019-03-26 13:08:02 +00:00
|
|
|
{
|
2019-07-17 06:38:48 +00:00
|
|
|
if (handle_shortcut(keyCode))
|
2019-03-26 13:08:02 +00:00
|
|
|
{
|
2019-07-17 06:38:48 +00:00
|
|
|
update_data();
|
2019-03-26 13:08:02 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (processed)
|
2019-07-17 10:43:27 +00:00
|
|
|
m_parent.set_as_dirty();
|
2019-03-26 13:08:02 +00:00
|
|
|
|
|
|
|
return processed;
|
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
bool GLGizmosManager::on_key(wxKeyEvent& evt)
|
2019-03-26 13:38:30 +00:00
|
|
|
{
|
|
|
|
const int keyCode = evt.GetKeyCode();
|
|
|
|
bool processed = false;
|
|
|
|
|
|
|
|
if (evt.GetEventType() == wxEVT_KEY_UP)
|
|
|
|
{
|
2019-11-08 13:05:56 +00:00
|
|
|
if (m_current == SlaSupports || m_current == Hollow)
|
2019-04-24 13:43:52 +00:00
|
|
|
{
|
2019-11-08 13:05:56 +00:00
|
|
|
bool is_editing = true;
|
|
|
|
bool is_rectangle_dragging = false;
|
|
|
|
|
|
|
|
if (m_current == SlaSupports) {
|
|
|
|
GLGizmoSlaSupports* gizmo = dynamic_cast<GLGizmoSlaSupports*>(get_current());
|
|
|
|
is_editing = gizmo->is_in_editing_mode();
|
|
|
|
is_rectangle_dragging = gizmo->is_selection_rectangle_dragging();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
GLGizmoHollow* gizmo = dynamic_cast<GLGizmoHollow*>(get_current());
|
|
|
|
is_rectangle_dragging = gizmo->is_selection_rectangle_dragging();
|
|
|
|
}
|
2019-04-24 13:43:52 +00:00
|
|
|
|
|
|
|
if (keyCode == WXK_SHIFT)
|
|
|
|
{
|
|
|
|
// shift has been just released - SLA gizmo might want to close rectangular selection.
|
2019-11-08 13:05:56 +00:00
|
|
|
if (gizmo_event(SLAGizmoEventType::ShiftUp) || (is_editing && is_rectangle_dragging))
|
2019-04-24 13:43:52 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
else if (keyCode == WXK_ALT)
|
|
|
|
{
|
|
|
|
// alt has been just released - SLA gizmo might want to close rectangular selection.
|
2019-11-08 13:05:56 +00:00
|
|
|
if (gizmo_event(SLAGizmoEventType::AltUp) || (is_editing && is_rectangle_dragging))
|
2019-04-24 13:43:52 +00:00
|
|
|
processed = true;
|
|
|
|
}
|
|
|
|
}
|
2020-02-03 15:05:13 +00:00
|
|
|
|
|
|
|
// if (processed)
|
|
|
|
// m_parent.set_cursor(GLCanvas3D::Standard);
|
|
|
|
}
|
|
|
|
else if (evt.GetEventType() == wxEVT_KEY_DOWN)
|
|
|
|
{
|
|
|
|
if ((m_current == SlaSupports) && ((keyCode == WXK_SHIFT) || (keyCode == WXK_ALT))
|
|
|
|
&& dynamic_cast<GLGizmoSlaSupports*>(get_current())->is_in_editing_mode())
|
|
|
|
{
|
|
|
|
// m_parent.set_cursor(GLCanvas3D::Cross);
|
|
|
|
processed = true;
|
|
|
|
}
|
2020-02-03 14:00:49 +00:00
|
|
|
else if (m_current == Cut)
|
|
|
|
{
|
|
|
|
auto do_move = [this, &processed](double delta_z) {
|
|
|
|
GLGizmoCut* cut = dynamic_cast<GLGizmoCut*>(get_current());
|
|
|
|
cut->set_cut_z(delta_z + cut->get_cut_z());
|
|
|
|
processed = true;
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (keyCode)
|
|
|
|
{
|
|
|
|
case WXK_NUMPAD_UP: case WXK_UP: { do_move(1.0); break; }
|
|
|
|
case WXK_NUMPAD_DOWN: case WXK_DOWN: { do_move(-1.0); break; }
|
|
|
|
default: { break; }
|
|
|
|
}
|
|
|
|
}
|
2019-03-26 13:38:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (processed)
|
2019-07-17 10:43:27 +00:00
|
|
|
m_parent.set_as_dirty();
|
2019-03-26 13:38:30 +00:00
|
|
|
|
|
|
|
return processed;
|
|
|
|
}
|
|
|
|
|
2019-08-22 14:44:49 +00:00
|
|
|
void GLGizmosManager::update_after_undo_redo(const UndoRedo::Snapshot& snapshot)
|
2019-07-17 06:38:48 +00:00
|
|
|
{
|
|
|
|
update_data();
|
|
|
|
m_serializing = false;
|
2019-08-22 14:44:49 +00:00
|
|
|
if (m_current == SlaSupports
|
|
|
|
&& snapshot.snapshot_data.flags & UndoRedo::SnapshotData::RECALCULATE_SLA_SUPPORTS)
|
2019-09-02 12:25:48 +00:00
|
|
|
dynamic_cast<GLGizmoSlaSupports*>(m_gizmos[SlaSupports].get())->reslice_SLA_supports(true);
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-07-20 10:02:29 +00:00
|
|
|
void GLGizmosManager::render_background(float left, float top, float right, float bottom, float border) const
|
|
|
|
{
|
|
|
|
unsigned int tex_id = m_background_texture.texture.get_id();
|
|
|
|
float tex_width = (float)m_background_texture.texture.get_width();
|
|
|
|
float tex_height = (float)m_background_texture.texture.get_height();
|
|
|
|
if ((tex_id != 0) && (tex_width > 0) && (tex_height > 0))
|
|
|
|
{
|
|
|
|
float inv_tex_width = (tex_width != 0.0f) ? 1.0f / tex_width : 0.0f;
|
|
|
|
float inv_tex_height = (tex_height != 0.0f) ? 1.0f / tex_height : 0.0f;
|
|
|
|
|
|
|
|
float internal_left = left + border;
|
|
|
|
float internal_right = right - border;
|
|
|
|
float internal_top = top - border;
|
|
|
|
float internal_bottom = bottom + border;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
// float left_uv = 0.0f;
|
2019-07-20 10:02:29 +00:00
|
|
|
float right_uv = 1.0f;
|
|
|
|
float top_uv = 1.0f;
|
|
|
|
float bottom_uv = 0.0f;
|
|
|
|
|
|
|
|
float internal_left_uv = (float)m_background_texture.metadata.left * inv_tex_width;
|
|
|
|
float internal_right_uv = 1.0f - (float)m_background_texture.metadata.right * inv_tex_width;
|
|
|
|
float internal_top_uv = 1.0f - (float)m_background_texture.metadata.top * inv_tex_height;
|
|
|
|
float internal_bottom_uv = (float)m_background_texture.metadata.bottom * inv_tex_height;
|
|
|
|
|
|
|
|
// top-left corner
|
|
|
|
GLTexture::render_sub_texture(tex_id, left, internal_left, internal_top, top, { { internal_left_uv, internal_bottom_uv }, { internal_right_uv, internal_bottom_uv }, { internal_right_uv, internal_top_uv }, { internal_left_uv, internal_top_uv } });
|
|
|
|
|
|
|
|
// top edge
|
|
|
|
GLTexture::render_sub_texture(tex_id, internal_left, internal_right, internal_top, top, { { internal_left_uv, internal_top_uv }, { internal_right_uv, internal_top_uv }, { internal_right_uv, top_uv }, { internal_left_uv, top_uv } });
|
|
|
|
|
|
|
|
// top-right corner
|
|
|
|
GLTexture::render_sub_texture(tex_id, internal_right, right, internal_top, top, { { internal_right_uv, internal_top_uv }, { right_uv, internal_top_uv }, { right_uv, top_uv }, { internal_right_uv, top_uv } });
|
|
|
|
|
|
|
|
// center-left edge
|
|
|
|
GLTexture::render_sub_texture(tex_id, left, internal_left, internal_bottom, internal_top, { { internal_left_uv, internal_bottom_uv }, { internal_right_uv, internal_bottom_uv }, { internal_right_uv, internal_top_uv }, { internal_left_uv, internal_top_uv } });
|
|
|
|
|
|
|
|
// center
|
|
|
|
GLTexture::render_sub_texture(tex_id, internal_left, internal_right, internal_bottom, internal_top, { { internal_left_uv, internal_bottom_uv }, { internal_right_uv, internal_bottom_uv }, { internal_right_uv, internal_top_uv }, { internal_left_uv, internal_top_uv } });
|
|
|
|
|
|
|
|
// center-right edge
|
|
|
|
GLTexture::render_sub_texture(tex_id, internal_right, right, internal_bottom, internal_top, { { internal_right_uv, internal_bottom_uv }, { right_uv, internal_bottom_uv }, { right_uv, internal_top_uv }, { internal_right_uv, internal_top_uv } });
|
|
|
|
|
|
|
|
// bottom-left corner
|
|
|
|
GLTexture::render_sub_texture(tex_id, left, internal_left, bottom, internal_bottom, { { internal_left_uv, internal_bottom_uv }, { internal_right_uv, internal_bottom_uv }, { internal_right_uv, internal_top_uv }, { internal_left_uv, internal_top_uv } });
|
|
|
|
|
|
|
|
// bottom edge
|
|
|
|
GLTexture::render_sub_texture(tex_id, internal_left, internal_right, bottom, internal_bottom, { { internal_left_uv, bottom_uv }, { internal_right_uv, bottom_uv }, { internal_right_uv, internal_bottom_uv }, { internal_left_uv, internal_bottom_uv } });
|
|
|
|
|
|
|
|
// bottom-right corner
|
|
|
|
GLTexture::render_sub_texture(tex_id, internal_right, right, bottom, internal_bottom, { { internal_right_uv, bottom_uv }, { right_uv, bottom_uv }, { right_uv, internal_bottom_uv }, { internal_right_uv, internal_bottom_uv } });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
void GLGizmosManager::do_render_overlay() const
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2020-01-07 11:40:03 +00:00
|
|
|
std::vector<size_t> selectable_idxs = get_selectable_idxs();
|
|
|
|
if (selectable_idxs.empty())
|
|
|
|
return;
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-07-17 10:43:27 +00:00
|
|
|
float cnv_w = (float)m_parent.get_canvas_size().get_width();
|
|
|
|
float cnv_h = (float)m_parent.get_canvas_size().get_height();
|
2020-03-02 09:58:46 +00:00
|
|
|
float zoom = (float)wxGetApp().plater()->get_camera().get_zoom();
|
|
|
|
float inv_zoom = (float)wxGetApp().plater()->get_camera().get_inv_zoom();
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-12-12 10:11:11 +00:00
|
|
|
float height = get_scaled_total_height();
|
|
|
|
float width = get_scaled_total_width();
|
|
|
|
float zoomed_border = m_layout.scaled_border() * inv_zoom;
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-12-12 10:11:11 +00:00
|
|
|
float zoomed_top_x = (-0.5f * cnv_w) * inv_zoom;
|
|
|
|
float zoomed_top_y = (0.5f * height) * inv_zoom;
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-12-12 10:11:11 +00:00
|
|
|
float zoomed_left = zoomed_top_x;
|
|
|
|
float zoomed_top = zoomed_top_y;
|
|
|
|
float zoomed_right = zoomed_left + width * inv_zoom;
|
|
|
|
float zoomed_bottom = zoomed_top - height * inv_zoom;
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-12-12 10:11:11 +00:00
|
|
|
render_background(zoomed_left, zoomed_top, zoomed_right, zoomed_bottom, zoomed_border);
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-12-12 10:11:11 +00:00
|
|
|
zoomed_top_x += zoomed_border;
|
|
|
|
zoomed_top_y -= zoomed_border;
|
|
|
|
|
|
|
|
float icons_size = m_layout.scaled_icons_size();
|
|
|
|
float zoomed_icons_size = icons_size * inv_zoom;
|
|
|
|
float zoomed_stride_y = m_layout.scaled_stride_y() * inv_zoom;
|
2019-03-20 12:51:25 +00:00
|
|
|
|
|
|
|
unsigned int icons_texture_id = m_icons_texture.get_id();
|
2019-09-02 12:25:48 +00:00
|
|
|
int tex_width = m_icons_texture.get_width();
|
|
|
|
int tex_height = m_icons_texture.get_height();
|
2019-07-12 08:13:35 +00:00
|
|
|
|
2020-01-07 11:40:03 +00:00
|
|
|
if ((icons_texture_id == 0) || (tex_width <= 1) || (tex_height <= 1))
|
|
|
|
return;
|
|
|
|
|
|
|
|
float du = (float)(tex_width - 1) / (4.0f * (float)tex_width); // 4 is the number of possible states if the icons
|
2020-01-08 09:04:39 +00:00
|
|
|
float dv = (float)(tex_height - 1) / (float)(m_gizmos.size() * tex_height);
|
2020-01-07 11:40:03 +00:00
|
|
|
|
|
|
|
// tiles in the texture are spaced by 1 pixel
|
|
|
|
float u_offset = 1.0f / (float)tex_width;
|
|
|
|
float v_offset = 1.0f / (float)tex_height;
|
|
|
|
|
|
|
|
for (size_t idx : selectable_idxs)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-09-02 12:25:48 +00:00
|
|
|
GLGizmoBase* gizmo = m_gizmos[idx].get();
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-08-19 09:36:55 +00:00
|
|
|
unsigned int sprite_id = gizmo->get_sprite_id();
|
2019-12-09 14:21:18 +00:00
|
|
|
int icon_idx = (m_current == idx) ? 2 : ((m_hover == idx) ? 1 : (gizmo->is_activable()? 0 : 3));
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2020-01-07 11:40:03 +00:00
|
|
|
float v_top = v_offset + sprite_id * dv;
|
|
|
|
float u_left = u_offset + icon_idx * du;
|
|
|
|
float v_bottom = v_top + dv - v_offset;
|
|
|
|
float u_right = u_left + du - u_offset;
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-12-12 10:11:11 +00:00
|
|
|
GLTexture::render_sub_texture(icons_texture_id, zoomed_top_x, zoomed_top_x + zoomed_icons_size, zoomed_top_y - zoomed_icons_size, zoomed_top_y, { { u_left, v_bottom }, { u_right, v_bottom }, { u_right, v_top }, { u_left, v_top } });
|
2019-08-19 10:11:26 +00:00
|
|
|
if (idx == m_current) {
|
2020-03-02 09:58:46 +00:00
|
|
|
float toolbar_top = cnv_h - wxGetApp().plater()->get_view_toolbar().get_height();
|
2019-12-12 10:11:11 +00:00
|
|
|
gizmo->render_input_window(width, 0.5f * cnv_h - zoomed_top_y * zoom, toolbar_top);
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
2019-12-12 10:11:11 +00:00
|
|
|
zoomed_top_y -= zoomed_stride_y;
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-12 10:11:11 +00:00
|
|
|
float GLGizmosManager::get_scaled_total_height() const
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-12-12 10:11:11 +00:00
|
|
|
return m_layout.scale * (2.0f * m_layout.border + (float)get_selectable_idxs().size() * m_layout.stride_y() - m_layout.gap_y);
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
2019-12-12 10:11:11 +00:00
|
|
|
float GLGizmosManager::get_scaled_total_width() const
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-12-12 10:11:11 +00:00
|
|
|
return 2.0f * m_layout.scaled_border() + m_layout.scaled_icons_size();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GLGizmoBase* GLGizmosManager::get_current() const
|
|
|
|
{
|
2019-12-12 10:11:11 +00:00
|
|
|
return ((m_current == Undefined) || m_gizmos.empty()) ? nullptr : m_gizmos[m_current].get();
|
2019-03-20 12:51:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool GLGizmosManager::generate_icons_texture() const
|
|
|
|
{
|
|
|
|
std::string path = resources_dir() + "/icons/";
|
|
|
|
std::vector<std::string> filenames;
|
2019-08-19 09:36:55 +00:00
|
|
|
for (size_t idx=0; idx<m_gizmos.size(); ++idx)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
if (m_gizmos[idx] != nullptr)
|
2019-03-20 12:51:25 +00:00
|
|
|
{
|
2019-08-19 09:36:55 +00:00
|
|
|
const std::string& icon_filename = m_gizmos[idx]->get_icon_filename();
|
2019-03-20 12:51:25 +00:00
|
|
|
if (!icon_filename.empty())
|
|
|
|
filenames.push_back(path + icon_filename);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<std::pair<int, bool>> states;
|
2019-12-09 14:21:18 +00:00
|
|
|
states.push_back(std::make_pair(1, false)); // Activable
|
|
|
|
states.push_back(std::make_pair(0, false)); // Hovered
|
|
|
|
states.push_back(std::make_pair(0, true)); // Selected
|
|
|
|
states.push_back(std::make_pair(2, false)); // Disabled
|
2019-03-20 12:51:25 +00:00
|
|
|
|
2019-12-12 10:11:11 +00:00
|
|
|
unsigned int sprite_size_px = (unsigned int)m_layout.scaled_icons_size();
|
2019-12-05 13:43:18 +00:00
|
|
|
// // force even size
|
|
|
|
// if (sprite_size_px % 2 != 0)
|
|
|
|
// sprite_size_px += 1;
|
2019-12-05 11:22:24 +00:00
|
|
|
|
|
|
|
bool res = m_icons_texture.load_from_svg_files_as_sprites_array(filenames, states, sprite_size_px, false);
|
2019-03-20 12:51:25 +00:00
|
|
|
if (res)
|
|
|
|
m_icons_texture_dirty = false;
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
void GLGizmosManager::update_on_off_state(const Vec2d& mouse_pos)
|
2019-03-26 11:30:17 +00:00
|
|
|
{
|
2019-07-17 10:43:27 +00:00
|
|
|
if (!m_enabled)
|
2019-03-26 11:30:17 +00:00
|
|
|
return;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
size_t idx = get_gizmo_idx_from_mouse(mouse_pos);
|
|
|
|
if (idx != Undefined && m_gizmos[idx]->is_activable() && m_hover == idx)
|
|
|
|
activate_gizmo(m_current == idx ? Undefined : (EType)idx);
|
2019-03-26 11:30:17 +00:00
|
|
|
}
|
|
|
|
|
2019-07-17 06:38:48 +00:00
|
|
|
std::string GLGizmosManager::update_hover_state(const Vec2d& mouse_pos)
|
2019-03-26 08:01:04 +00:00
|
|
|
{
|
|
|
|
std::string name = "";
|
|
|
|
|
2019-07-17 10:43:27 +00:00
|
|
|
if (!m_enabled)
|
2019-03-26 08:01:04 +00:00
|
|
|
return name;
|
|
|
|
|
2019-08-19 09:36:55 +00:00
|
|
|
m_hover = Undefined;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
size_t idx = get_gizmo_idx_from_mouse(mouse_pos);
|
|
|
|
if (idx != Undefined) {
|
|
|
|
name = m_gizmos[idx]->get_name();
|
2019-03-26 08:01:04 +00:00
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
if (m_gizmos[idx]->is_activable())
|
2019-08-19 09:36:55 +00:00
|
|
|
m_hover = (EType)idx;
|
2019-03-26 08:01:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
2019-08-19 10:11:26 +00:00
|
|
|
void GLGizmosManager::activate_gizmo(EType type)
|
|
|
|
{
|
|
|
|
if (m_gizmos.empty() || m_current == type)
|
|
|
|
return;
|
|
|
|
|
2019-09-02 12:25:48 +00:00
|
|
|
if (m_current != Undefined) {
|
2019-08-19 10:11:26 +00:00
|
|
|
m_gizmos[m_current]->set_state(GLGizmoBase::Off);
|
2019-09-02 12:25:48 +00:00
|
|
|
if (m_gizmos[m_current]->get_state() != GLGizmoBase::Off)
|
|
|
|
return; // gizmo refused to be turned off, do nothing.
|
|
|
|
}
|
2019-08-19 10:11:26 +00:00
|
|
|
|
2020-04-03 16:04:37 +00:00
|
|
|
m_current = type;
|
2020-06-01 08:01:45 +00:00
|
|
|
|
|
|
|
// Updating common data should be left to the update_data function, which
|
|
|
|
// is always called after this one. activate_gizmo can be called by undo/redo,
|
|
|
|
// when selection is not yet deserialized, so the common data would update
|
|
|
|
// incorrectly (or crash if relying on unempty selection). Undo/redo stack
|
|
|
|
// will also call update_data, after selection is restored.
|
|
|
|
|
|
|
|
//m_common_gizmos_data->update(get_current()
|
|
|
|
// ? get_current()->get_requirements()
|
|
|
|
// : CommonGizmosDataID(0));
|
2020-04-03 16:04:37 +00:00
|
|
|
|
2019-08-19 10:11:26 +00:00
|
|
|
if (type != Undefined)
|
|
|
|
m_gizmos[type]->set_state(GLGizmoBase::On);
|
|
|
|
}
|
|
|
|
|
2019-03-26 08:01:04 +00:00
|
|
|
|
2019-03-26 11:30:17 +00:00
|
|
|
bool GLGizmosManager::grabber_contains_mouse() const
|
|
|
|
{
|
|
|
|
if (!m_enabled)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
GLGizmoBase* curr = get_current();
|
|
|
|
return (curr != nullptr) ? (curr->get_hover_id() != -1) : false;
|
|
|
|
}
|
|
|
|
|
2019-03-20 12:51:25 +00:00
|
|
|
} // namespace GUI
|
|
|
|
} // namespace Slic3r
|