mirror of
https://git.mirrors.martin98.com/https://github.com/SoftFever/OrcaSlicer.git
synced 2025-07-31 01:06:47 +08:00

* NEW:add move and rotate gizmo in assemble view Cherry-picked from bambulab/BambuStudio@d9e47bd9a9 Co-authored-by: zhou.xu <zhou.xu@bambulab.com> * Deselect other parts if Alt is pressed when selecting Cherry-picked from bambulab/BambuStudio@f5eb2899e7 Co-authored-by: zhou.xu <zhou.xu@bambulab.com> * FIX:z offset is error after copy and paste several objects jira: STUDIO-6753 STUDIO-7135 Change-Id: I6d9c8eb0c957ff1e3194709704ceb6c3920baa4f (cherry picked from commit 847a7141a6f47e409566b19e73c0ebdeb08f39e2) (cherry picked from commit a5cc52beb7eef5848368e660ca4f14e95ad5f7d5) * FIX:arrow direction in scaling tool is incorrect Jira: STUDIO-5672 Change-Id: I82c0ab336805e34c8380f93e64d3b9dbbf283805 (cherry picked from commit f6f27b700f0305854fcdbcb1191af25a4b8bdbe4) * FIX:world cs is displayed incorrectly The value of world coordinate system for model_volume is displayed incorrectly Jira: STUDIO-6399 code is from PrusaSlicer thanks for PrusaSlicer and enricoturri1966 commit 325709c5ae9b937867b36103a41d12a102c99292 Author: enricoturri1966 <enricoturri@seznam.cz> Date: Thu Jan 26 15:49:00 2023 +0100 SPE-1419 - Fixed reset skew resetting mirror, reset scale resetting mirror, changed labels in Object Manipulator panel, scale of instances using the Object Manipulator panel always made as absolute Change-Id: I30fdd39effd73b8dc027e4263fa7e64937b84326 Cherry-picked from bambulab/BambuStudio@0b46b9848b Co-authored-by: enricoturri1966 <enricoturri@seznam.cz> * FIX:fix scale problem add tool tip for move,rotate,scale gizmo Jira: STUDIO-6425 STUDIO-6419 Change-Id: I0b89c9b70f83cde21c6a407bcecd78c925515cfa Cherry-picked from bambulab/BambuStudio@6dad59102b Co-authored-by: zhou.xu <zhou.xu@bambulab.com> * NEW:add Object coordinates in move gizmo jira: none Part of the code references PrusaSlicer,thanks for PrusaSlicer and enricoturri1966 commit c12eeee12f9e2c91a9dfe7905f1370143805f038 Author: enricoturri1966 <enricoturri@seznam.cz> Date: Mon Oct 2 14:26:36 2023 +0200 SPE-1926: Tech ENABLE_CGAL_BOUNDING_SPHERE - Use selection's bounding sphere center as pivot for rotations Change-Id: Iae7e4539c198af3ff1aa99e1c0ce015fbcf80256 (cherry picked from commit 2b73bc915ee27218c9803ba0a01b0d3e47adf1da) Cherry-picked from bambulab/BambuStudio@98cce3b656 Co-authored-by: enricoturri1966 <enricoturri@seznam.cz> Co-authored-by: zhou.xu <zhou.xu@bambulab.com> * FIX:fix imgui style at Object coordinate in move tool jira:STUDIO-7141 Change-Id: Ib2900012c28878c4e7ad97eb0cf319f693cb9f6f (cherry picked from commit b7b09c82897678c4f3615713bc5d1cc7a3b17b19) (cherry picked from commit c89732a04619a6d910b723c126515bae802f7167) * ENH:use local cs for non_model_part better jira: STUDIO-7234 Change-Id: I0f0e99429e5e0b7cc4932a661eceffcff4a495f6 (cherry picked from commit b4305a3bfc9e5ae05c1785a710238a70f2dfb44a) (cherry picked from commit b28ac4f812f0024ec619c5d1b3c96e4cef4debdb) * ENH:add a cross mark for object cs jira: STUDIO-6947 Change-Id: Iaaab4f072045756ac3ba12c3f65e1c7f04ba65b8 (cherry picked from commit a2a2f49b4d94f257d36c9d17b4ec952e5dc9f0eb) Cherry-picked from bambulab/BambuStudio@8400e162a7 Co-authored-by: zhou.xu <zhou.xu@bambulab.com> * NEW:add tip button for move,rotate,scale jira: STUDIO-7273 Change-Id: I44aeecd8aaa17ec49ac1d8ff2bee5c3729c52061 (cherry picked from commit 998f33b4ce588f59cef345e327a97f6f669f6089) (cherry picked from commit f5eb2899e7252ea3ff0f8a79ef8d55c6009ebb28) * FIX:scale and size sholud >0 in scale tool jira: STUDIO-7433 Change-Id: Ibd4d00d9ca4762d002049e97a6d0819649f464db (cherry picked from commit eaaf11031ee49009af14abbd05bb4a07c88aceda) (cherry picked from commit 0d393d64b804ba7ae05454bf158de470cc74a6a6) * Fix crossmark rendering * Use combox as coord selection Cherry-picked from bambulab/BambuStudio@56f628dac1 Co-authored-by: zhou.xu <zhou.xu@bambulab.com> * NEW:add "world coordinates" scale for scale gizmo upgrade Transformation class jira:none about 75% code is from PrusaSlicer,thanks for PrusaSlicer and enricoturri1966 commit b32e9366606dce7d4f8de8db84fd902113bdbe28 Author: enricoturri1966 <enricoturri@seznam.cz> Date: Tue Mar 7 14:32:18 2023 +0100 Rework of constrained scaling Change-Id: I1248ea586e6b8f2fb6cdf3aa901ed7f525c3f111 (cherry picked from commit e10381aad1412b0c47afa340b634faa3af9d1a1f) Cherry-picked from bambulab/BambuStudio@c0536c09b4 Co-authored-by: enricoturri1966 <enricoturri@seznam.cz> Co-authored-by: zhou.xu <zhou.xu@bambulab.com> * ENH:set "Rotate (relative)" jira:none code is from PrusaSlicer,thanks for PrusaSlicer and enricoturri1966 commit 243985173e70c189ad9a86eefaaea0757d9749cb Author: enricoturri1966 <enricoturri@seznam.cz> Date: Thu May 12 14:33:41 2022 +0200 Tech ENABLE_TRANSFORMATIONS_BY_MATRICES - Allow for relative rotations Change-Id: I851939093ffb6881542fb21b434e17cc31a6dab2 (cherry picked from commit e412fa3492fa2ef59b84a84be1ede80935fb8a8d) * FIX:limit scaling ratio by grabber in scale tool jira: none Change-Id: I20a4404d4e4025ae230ab46ba8d8d3e5ffed10e3 (cherry picked from commit 97f63f167e80e859fec49666c8986f5a01f61838) * FIX:selection should be not empty when update_ui_from_settings jira: none Change-Id: I74b76733eba03d148dfd70279ec2ba65f19cc39a (cherry picked from commit f402685aee747fe5c3740b2cb80fc2a60e129918) * ENH:add "volume selection" checkbox jira: none Change-Id: I68b5f54e37ea2ab9e2b65ac84abc834060f400df (cherry picked from commit eec7de441bd40408fe688587d2834b0c42c0d66f) * FIX:add can_sequential_clearance_show_in_gizmo api jira: STUDIO-7836 Change-Id: Ie0cded272596bafee4e491e379722dcc23035dc4 (cherry picked from commit 715d2b9b7840939663e99e0ecbfcefd8ecf2904f) * FIX:select all should ban in paint,cut and so on gizmo jira: STUDIO-7872 Change-Id: Ic6496dbdd892814e1fc41625ee34ffc46f171657 (cherry picked from commit 95e8ca728553081db4ecbb3d865c8b999a6ff2fa) * FIX:add wipe tower'position in move gizmo jira: STUDIO-7861 Change-Id: I8147717bc61ba06a7e1fba45532cdadc2ba1174e (cherry picked from commit 065dddb890d3ec81643b9767397bdad72ae69ebd) * ENH:fix text coordinate system calculation jira: STUDIO-6449 Change-Id: I36214c14c348e8f52b96501cd027205819b0dabc (cherry picked from commit 44287812a0cb212f1bf6fe70e32e1075f532886d) Cherry-picked from bambulab/BambuStudio@4091f3e042 Co-authored-by: zhou.xu <zhou.xu@bambulab.com> * FIX:delete old selection.translate api jira: STUDIO-8201 code is from PrusaSlicer,thanks for PrusaSlicer and enricoturri1966 commit 88ce6ccdef5f680709ea8b676688784a7af287dd Author: enricoturri1966 <enricoturri@seznam.cz> Date: Wed May 11 10:54:42 2022 +0200 Tech ENABLE_TRANSFORMATIONS_BY_MATRICES - Change-Id: Iafe963f0f7bf9028f32a4fb4a4cc8cc609662283 Change-Id: Ibbc36c004734f35564f0028dd1e537ac926a2f1f Cherry-picked from bambulab/BambuStudio@c6d9f2685e Co-authored-by: enricoturri1966 <enricoturri@seznam.cz> Co-authored-by: zhou.xu <zhou.xu@bambulab.com> * FIX:add protection for null pointer jira: none Change-Id: I9a9231bab893f5d2afa008f65165269ae176c962 (cherry picked from commit f27a713aaf77b1109fc57b8650efa6b23081f799) * FIX:when two dir is perpendicular to each other,scale error (plane_normal.dot(ray_dir)) jira:STUDIO-8274 Change-Id: Ib3145ab75e18c832d20065d204aa41b75f73b673 (cherry picked from commit fbdc9cd580f835d1a873d08ed64baed3b3db6f9a) * ENH:add "reset real zeros" button in rotate gizmo jira: STUDIO-8291 Change-Id: Ia10e4d8a2a3a073c22a1306aeab9ffa3e7b77c2b (cherry picked from commit 738e3f004daa9082709800e4e3d0d9bbe1b7ed7e) * FIX:add "absolute rotation" in rotate gizmo jira: STUDIO-8726 Change-Id: I23deb4ab11cf24ca4f0f0c5a35a74268c34f60f6 (cherry picked from commit d26b8f9fcadf8f7709a302991e43be711560e84e) (cherry picked from commit 496d69f9d1b91c6bd84804e57a276bccf79f0cbd) * Fix tooltip button size * Fix issue that reative rotation history not cleared after gizmo closed * Show selection box in assemble view * ENH:add an tip icon for assembly view jira: STUDIO-7155 Change-Id: Ie9e4fa578c8aa5bda9ff771d82f396f8b51026bb (cherry picked from commit 515f9473347fb912a9dc8c365f1c318506096083) --------- Co-authored-by: zhou.xu <zhou.xu@bambulab.com> Co-authored-by: enricoturri1966 <enricoturri@seznam.cz> Co-authored-by: SoftFever <softfeverever@gmail.com>
473 lines
18 KiB
C++
473 lines
18 KiB
C++
#include "GLGizmoScale.hpp"
|
|
#include "slic3r/GUI/GLCanvas3D.hpp"
|
|
#include "slic3r/GUI/GUI_App.hpp"
|
|
#include "slic3r/GUI/Plater.hpp"
|
|
|
|
#include <GL/glew.h>
|
|
|
|
#include <wx/utils.h>
|
|
|
|
namespace Slic3r {
|
|
namespace GUI {
|
|
|
|
|
|
const float GLGizmoScale3D::Offset = 5.0f;
|
|
|
|
// get intersection of ray and plane
|
|
Vec3d GetIntersectionOfRayAndPlane(Vec3d ray_position, Vec3d ray_dir, Vec3d plane_position, Vec3d plane_normal)
|
|
{
|
|
double t = (plane_normal.dot(plane_position) - plane_normal.dot(ray_position)) / (plane_normal.dot(ray_dir));
|
|
Vec3d intersection = ray_position + t * ray_dir;
|
|
return intersection;
|
|
}
|
|
|
|
//BBS: GUI refactor: add obj manipulation
|
|
GLGizmoScale3D::GLGizmoScale3D(GLCanvas3D& parent, const std::string& icon_filename, unsigned int sprite_id, GizmoObjectManipulation* obj_manipulation)
|
|
: GLGizmoBase(parent, icon_filename, sprite_id)
|
|
, m_scale(Vec3d::Ones())
|
|
, m_offset(Vec3d::Zero())
|
|
, m_snap_step(0.05)
|
|
//BBS: GUI refactor: add obj manipulation
|
|
, m_object_manipulation(obj_manipulation)
|
|
{
|
|
m_grabber_connections[0].grabber_indices = { 0, 1 };
|
|
m_grabber_connections[1].grabber_indices = { 2, 3 };
|
|
m_grabber_connections[2].grabber_indices = { 4, 5 };
|
|
m_grabber_connections[3].grabber_indices = { 6, 7 };
|
|
m_grabber_connections[4].grabber_indices = { 7, 8 };
|
|
m_grabber_connections[5].grabber_indices = { 8, 9 };
|
|
m_grabber_connections[6].grabber_indices = { 9, 6 };
|
|
}
|
|
|
|
const Vec3d &GLGizmoScale3D::get_scale()
|
|
{
|
|
if (m_object_manipulation) {
|
|
Vec3d cache_scale = m_object_manipulation->get_cache().scale.cwiseQuotient(Vec3d(100,100,100));
|
|
Vec3d temp_scale = cache_scale.cwiseProduct(m_scale);
|
|
m_object_manipulation->limit_scaling_ratio(temp_scale);
|
|
m_scale = temp_scale.cwiseQuotient(cache_scale);
|
|
}
|
|
return m_scale;
|
|
}
|
|
|
|
std::string GLGizmoScale3D::get_tooltip() const
|
|
{
|
|
const Selection& selection = m_parent.get_selection();
|
|
|
|
bool single_instance = selection.is_single_full_instance();
|
|
bool single_volume = selection.is_single_volume_or_modifier();
|
|
|
|
Vec3f scale = 100.0f * Vec3f::Ones();
|
|
if (single_instance)
|
|
scale = 100.0f * selection.get_first_volume()->get_instance_scaling_factor().cast<float>();
|
|
else if (single_volume)
|
|
scale = 100.0f * selection.get_first_volume()->get_volume_scaling_factor().cast<float>();
|
|
|
|
if (m_hover_id == 0 || m_hover_id == 1 || m_grabbers[0].dragging || m_grabbers[1].dragging)
|
|
return "X: " + format(scale.x(), 4) + "%";
|
|
else if (m_hover_id == 2 || m_hover_id == 3 || m_grabbers[2].dragging || m_grabbers[3].dragging)
|
|
return "Y: " + format(scale.y(), 4) + "%";
|
|
else if (m_hover_id == 4 || m_hover_id == 5 || m_grabbers[4].dragging || m_grabbers[5].dragging)
|
|
return "Z: " + format(scale.z(), 4) + "%";
|
|
else if (m_hover_id == 6 || m_hover_id == 7 || m_hover_id == 8 || m_hover_id == 9 ||
|
|
m_grabbers[6].dragging || m_grabbers[7].dragging || m_grabbers[8].dragging || m_grabbers[9].dragging)
|
|
{
|
|
std::string tooltip = "X: " + format(scale.x(), 2) + "%\n";
|
|
tooltip += "Y: " + format(scale.y(), 2) + "%\n";
|
|
tooltip += "Z: " + format(scale.z(), 2) + "%";
|
|
return tooltip;
|
|
}
|
|
else
|
|
return "";
|
|
}
|
|
|
|
bool GLGizmoScale3D::on_mouse(const wxMouseEvent &mouse_event)
|
|
{
|
|
if (mouse_event.Dragging()) {
|
|
if (m_dragging) {
|
|
// Apply new temporary scale factors
|
|
TransformationType transformation_type;
|
|
if (wxGetApp().obj_manipul()->is_local_coordinates())
|
|
transformation_type.set_local();
|
|
else if (wxGetApp().obj_manipul()->is_instance_coordinates())
|
|
transformation_type.set_instance();
|
|
|
|
transformation_type.set_relative();
|
|
|
|
if (mouse_event.AltDown())
|
|
transformation_type.set_independent();
|
|
|
|
Selection& selection = m_parent.get_selection();
|
|
selection.scale_and_translate(get_scale(), get_offset(), transformation_type);
|
|
}
|
|
}
|
|
return use_grabbers(mouse_event);
|
|
}
|
|
|
|
void GLGizmoScale3D::data_changed(bool is_serializing)
|
|
{
|
|
const Selection &selection = m_parent.get_selection();
|
|
bool enable_scale_xyz = selection.is_single_full_instance() ||
|
|
selection.is_single_volume_or_modifier();
|
|
for (unsigned int i = 0; i < 6; ++i)
|
|
m_grabbers[i].enabled = enable_scale_xyz;
|
|
|
|
set_scale(Vec3d::Ones());
|
|
|
|
change_cs_by_selection();
|
|
}
|
|
|
|
void GLGizmoScale3D::enable_ununiversal_scale(bool enable)
|
|
{
|
|
for (unsigned int i = 0; i < 6; ++i)
|
|
m_grabbers[i].enabled = enable;
|
|
}
|
|
|
|
bool GLGizmoScale3D::on_init()
|
|
{
|
|
for (int i = 0; i < 10; ++i)
|
|
{
|
|
m_grabbers.push_back(Grabber());
|
|
}
|
|
|
|
double half_pi = 0.5 * (double)PI;
|
|
|
|
|
|
// BBS
|
|
m_grabbers[4].enabled = false;
|
|
m_shortcut_key = WXK_CONTROL_S;
|
|
|
|
return true;
|
|
}
|
|
|
|
std::string GLGizmoScale3D::on_get_name() const
|
|
{
|
|
if (!on_is_activable() && m_state == EState::Off) {
|
|
return _u8L("Scale") + ":\n" + _u8L("Please select at least one object.");
|
|
} else {
|
|
return _u8L("Scale");
|
|
}
|
|
}
|
|
|
|
bool GLGizmoScale3D::on_is_activable() const
|
|
{
|
|
const Selection& selection = m_parent.get_selection();
|
|
return !selection.is_empty() && !selection.is_wipe_tower();
|
|
}
|
|
|
|
void GLGizmoScale3D::on_set_state() {
|
|
if (get_state() == On) {
|
|
m_last_selected_obejct_idx = -1;
|
|
m_last_selected_volume_idx = -1;
|
|
change_cs_by_selection();
|
|
}
|
|
}
|
|
|
|
static int constraint_id(int grabber_id)
|
|
{
|
|
static const std::vector<int> id_map = {1, 0, 3, 2, 5, 4, 8, 9, 6, 7};
|
|
return (0 <= grabber_id && grabber_id < (int) id_map.size()) ? id_map[grabber_id] : -1;
|
|
}
|
|
|
|
void GLGizmoScale3D::on_start_dragging()
|
|
{
|
|
if (m_hover_id != -1) {
|
|
auto grabbers_transform = m_grabbers_tran.get_matrix();
|
|
m_starting.drag_position = grabbers_transform * m_grabbers[m_hover_id].center;
|
|
m_starting.plane_center = grabbers_transform * m_grabbers[4].center; // plane_center = bottom center
|
|
m_starting.plane_nromal = (grabbers_transform * m_grabbers[5].center - grabbers_transform * m_grabbers[4].center).normalized();
|
|
m_starting.ctrl_down = wxGetKeyState(WXK_CONTROL);
|
|
m_starting.box = m_bounding_box;
|
|
|
|
m_starting.center = m_center;
|
|
m_starting.instance_center = m_instance_center;
|
|
|
|
const Vec3d box_half_size = 0.5 * m_bounding_box.size();
|
|
|
|
m_starting.local_pivots[0] = Vec3d(box_half_size.x(), 0.0, -box_half_size.z());
|
|
m_starting.local_pivots[1] = Vec3d(-box_half_size.x(), 0.0, -box_half_size.z());
|
|
m_starting.local_pivots[2] = Vec3d(0.0, box_half_size.y(), -box_half_size.z());
|
|
m_starting.local_pivots[3] = Vec3d(0.0, -box_half_size.y(), -box_half_size.z());
|
|
m_starting.local_pivots[4] = Vec3d(0.0, 0.0, box_half_size.z());
|
|
m_starting.local_pivots[5] = Vec3d(0.0, 0.0, -box_half_size.z());
|
|
for (size_t i = 0; i < 6; i++) {
|
|
m_starting.pivots[i] = grabbers_transform * m_starting.local_pivots[i]; // todo delete
|
|
}
|
|
m_starting.constraint_position = grabbers_transform * m_grabbers[constraint_id(m_hover_id)].center;
|
|
m_scale = m_starting.scale = Vec3d::Ones() ;
|
|
m_offset = Vec3d::Zero();
|
|
}
|
|
}
|
|
|
|
void GLGizmoScale3D::on_stop_dragging()
|
|
{
|
|
m_parent.do_scale(L("Gizmo-Scale"));
|
|
m_starting.ctrl_down = false;
|
|
}
|
|
|
|
void GLGizmoScale3D::on_dragging(const UpdateData& data)
|
|
{
|
|
if ((m_hover_id == 0) || (m_hover_id == 1))
|
|
do_scale_along_axis(X, data);
|
|
else if ((m_hover_id == 2) || (m_hover_id == 3))
|
|
do_scale_along_axis(Y, data);
|
|
else if ((m_hover_id == 4) || (m_hover_id == 5))
|
|
do_scale_along_axis(Z, data);
|
|
else if (m_hover_id >= 6)
|
|
do_scale_uniform(data);
|
|
}
|
|
|
|
void GLGizmoScale3D::update_grabbers_data()
|
|
{
|
|
const Selection &selection = m_parent.get_selection();
|
|
const auto &[box, box_trafo] = selection.get_bounding_box_in_current_reference_system();
|
|
m_bounding_box = box;
|
|
m_center = box_trafo.translation();
|
|
m_grabbers_tran.set_matrix(box_trafo);
|
|
m_instance_center = (selection.is_single_full_instance() || selection.is_single_volume_or_modifier()) ? selection.get_first_volume()->get_instance_offset() : m_center;
|
|
|
|
const Vec3d box_half_size = 0.5 * m_bounding_box.size();
|
|
bool ctrl_down = wxGetKeyState(WXK_CONTROL);
|
|
|
|
|
|
bool single_instance = selection.is_single_full_instance();
|
|
bool single_volume = selection.is_single_modifier() || selection.is_single_volume();
|
|
|
|
// x axis
|
|
m_grabbers[0].center = Vec3d(-(box_half_size.x()), 0.0, -box_half_size.z());
|
|
m_grabbers[0].color = (ctrl_down && m_hover_id == 1) ? CONSTRAINED_COLOR : AXES_COLOR[0];
|
|
m_grabbers[1].center = Vec3d(box_half_size.x(), 0.0, -box_half_size.z());
|
|
m_grabbers[1].color = (ctrl_down && m_hover_id == 0) ? CONSTRAINED_COLOR : AXES_COLOR[0];
|
|
// y axis
|
|
m_grabbers[2].center = Vec3d(0.0, -(box_half_size.y()), -box_half_size.z());
|
|
m_grabbers[2].color = (ctrl_down && m_hover_id == 3) ? CONSTRAINED_COLOR : AXES_COLOR[1];
|
|
m_grabbers[3].center = Vec3d(0.0, box_half_size.y(), -box_half_size.z());
|
|
m_grabbers[3].color = (ctrl_down && m_hover_id == 2) ? CONSTRAINED_COLOR : AXES_COLOR[1];
|
|
// z axis do not show 4
|
|
m_grabbers[4].center = Vec3d(0.0, 0.0, -(box_half_size.z()));
|
|
m_grabbers[4].enabled = false;
|
|
|
|
m_grabbers[5].center = Vec3d(0.0, 0.0, box_half_size.z());
|
|
m_grabbers[5].color = (ctrl_down && m_hover_id == 4) ? CONSTRAINED_COLOR : AXES_COLOR[2];
|
|
// uniform
|
|
m_grabbers[6].center = Vec3d(-box_half_size.x(), -box_half_size.y(), -box_half_size.z());
|
|
m_grabbers[6].color = (ctrl_down && m_hover_id == 8) ? CONSTRAINED_COLOR : GRABBER_UNIFORM_COL;
|
|
m_grabbers[7].center = Vec3d(box_half_size.x(), -box_half_size.y(), -box_half_size.z());
|
|
m_grabbers[7].color = (ctrl_down && m_hover_id == 9) ? CONSTRAINED_COLOR : GRABBER_UNIFORM_COL;
|
|
m_grabbers[8].center = Vec3d(box_half_size.x(), box_half_size.y(), -box_half_size.z());
|
|
m_grabbers[8].color = (ctrl_down && m_hover_id == 6) ? CONSTRAINED_COLOR : GRABBER_UNIFORM_COL;
|
|
m_grabbers[9].center = Vec3d(-box_half_size.x(), box_half_size.y(), -box_half_size.z());
|
|
m_grabbers[9].color = (ctrl_down && m_hover_id == 7) ? CONSTRAINED_COLOR : GRABBER_UNIFORM_COL;
|
|
for (int i = 0; i < 6; ++i) {
|
|
//m_grabbers[i].color = AXES_COLOR[i / 2];
|
|
m_grabbers[i].hover_color = AXES_HOVER_COLOR[i / 2];
|
|
}
|
|
for (int i = 6; i < 10; ++i) {
|
|
//m_grabbers[i].color = GRABBER_UNIFORM_COL;
|
|
m_grabbers[i].hover_color = GRABBER_UNIFORM_HOVER_COL;
|
|
}
|
|
|
|
for (int i = 0; i < 10; ++i) {
|
|
m_grabbers[i].matrix = m_grabbers_tran.get_matrix();
|
|
}
|
|
}
|
|
|
|
|
|
void GLGizmoScale3D::change_cs_by_selection() {
|
|
int obejct_idx, volume_idx;
|
|
ModelVolume *model_volume = m_parent.get_selection().get_selected_single_volume(obejct_idx, volume_idx);
|
|
if (m_last_selected_obejct_idx == obejct_idx && m_last_selected_volume_idx == volume_idx) { return; }
|
|
m_last_selected_obejct_idx = obejct_idx;
|
|
m_last_selected_volume_idx = volume_idx;
|
|
if (m_parent.get_selection().is_multiple_full_object()) {
|
|
m_object_manipulation->set_coordinates_type(ECoordinatesType::World);
|
|
} else if (model_volume) {
|
|
m_object_manipulation->set_coordinates_type(ECoordinatesType::Local);
|
|
}
|
|
}
|
|
|
|
void GLGizmoScale3D::on_render()
|
|
{
|
|
glsafe(::glClear(GL_DEPTH_BUFFER_BIT));
|
|
glsafe(::glEnable(GL_DEPTH_TEST));
|
|
|
|
update_grabbers_data();
|
|
|
|
glsafe(::glLineWidth((m_hover_id != -1) ? 2.0f : 1.5f));
|
|
|
|
const float grabber_mean_size = (float)((m_bounding_box.size().x() + m_bounding_box.size().y() + m_bounding_box.size().z()) / 3.0);
|
|
|
|
//draw connections
|
|
GLShaderProgram* shader = wxGetApp().get_shader("flat");
|
|
if (shader != nullptr) {
|
|
shader->start_using();
|
|
// BBS: when select multiple objects, uniform scale can be deselected, display the connection(4,5)
|
|
//if (single_instance || single_volume) {
|
|
const Camera& camera = wxGetApp().plater()->get_camera();
|
|
shader->set_uniform("view_model_matrix", camera.get_view_matrix() * m_grabbers_tran.get_matrix());
|
|
shader->set_uniform("projection_matrix", camera.get_projection_matrix());
|
|
if (m_grabbers[4].enabled && m_grabbers[5].enabled)
|
|
render_grabbers_connection(4, 5, m_grabbers[4].color);
|
|
render_grabbers_connection(6, 7, m_grabbers[2].color);
|
|
render_grabbers_connection(7, 8, m_grabbers[2].color);
|
|
render_grabbers_connection(8, 9, m_grabbers[0].color);
|
|
render_grabbers_connection(9, 6, m_grabbers[0].color);
|
|
shader->stop_using();
|
|
}
|
|
|
|
// draw grabbers
|
|
shader = wxGetApp().get_shader("gouraud_light");
|
|
if (shader != nullptr) {
|
|
shader->start_using();
|
|
shader->set_uniform("emission_factor", 0.1f);
|
|
render_grabbers(grabber_mean_size);
|
|
shader->stop_using();
|
|
}
|
|
}
|
|
|
|
void GLGizmoScale3D::on_register_raycasters_for_picking()
|
|
{
|
|
// the gizmo grabbers are rendered on top of the scene, so the raytraced picker should take it into account
|
|
m_parent.set_raycaster_gizmos_on_top(true);
|
|
}
|
|
|
|
void GLGizmoScale3D::on_unregister_raycasters_for_picking()
|
|
{
|
|
m_parent.set_raycaster_gizmos_on_top(false);
|
|
}
|
|
|
|
void GLGizmoScale3D::render_grabbers_connection(unsigned int id_1, unsigned int id_2, const ColorRGBA& color)
|
|
{
|
|
auto grabber_connection = [this](unsigned int id_1, unsigned int id_2) {
|
|
for (int i = 0; i < int(m_grabber_connections.size()); ++i) {
|
|
if (m_grabber_connections[i].grabber_indices.first == id_1 && m_grabber_connections[i].grabber_indices.second == id_2)
|
|
return i;
|
|
}
|
|
return -1;
|
|
};
|
|
|
|
const int id = grabber_connection(id_1, id_2);
|
|
if (id == -1)
|
|
return;
|
|
|
|
if (!m_grabber_connections[id].model.is_initialized() ||
|
|
!m_grabber_connections[id].old_v1.isApprox(m_grabbers[id_1].center) ||
|
|
!m_grabber_connections[id].old_v2.isApprox(m_grabbers[id_2].center)) {
|
|
m_grabber_connections[id].old_v1 = m_grabbers[id_1].center;
|
|
m_grabber_connections[id].old_v2 = m_grabbers[id_2].center;
|
|
m_grabber_connections[id].model.reset();
|
|
|
|
GLModel::Geometry init_data;
|
|
init_data.format = { GLModel::Geometry::EPrimitiveType::Lines, GLModel::Geometry::EVertexLayout::P3 };
|
|
init_data.reserve_vertices(2);
|
|
init_data.reserve_indices(2);
|
|
|
|
// vertices
|
|
init_data.add_vertex((Vec3f)m_grabbers[id_1].center.cast<float>());
|
|
init_data.add_vertex((Vec3f)m_grabbers[id_2].center.cast<float>());
|
|
|
|
// indices
|
|
init_data.add_line(0, 1);
|
|
|
|
m_grabber_connections[id].model.init_from(std::move(init_data));
|
|
}
|
|
|
|
m_grabber_connections[id].model.set_color(color);
|
|
glLineStipple(1, 0x0FFF);
|
|
glEnable(GL_LINE_STIPPLE);
|
|
m_grabber_connections[id].model.render();
|
|
glDisable(GL_LINE_STIPPLE);
|
|
}
|
|
|
|
//BBS: add input window for move
|
|
void GLGizmoScale3D::on_render_input_window(float x, float y, float bottom_limit)
|
|
{
|
|
if (m_object_manipulation)
|
|
m_object_manipulation->do_render_scale_input_window(m_imgui, "Scale", x, y, bottom_limit);
|
|
}
|
|
|
|
void GLGizmoScale3D::do_scale_along_axis(Axis axis, const UpdateData& data)
|
|
{
|
|
double ratio = calc_ratio(data);
|
|
if (ratio > 0.0)
|
|
{
|
|
m_scale(axis) = m_starting.scale(axis) * ratio;
|
|
if (m_starting.ctrl_down && abs(ratio-1.0f)>0.001) {
|
|
double local_offset = 0.5 * (m_scale(axis) - m_starting.scale(axis)) * m_starting.box.size()(axis);
|
|
if (m_hover_id == 2 * axis) {
|
|
local_offset *= -1.0;
|
|
}
|
|
Vec3d local_offset_vec;
|
|
switch (axis)
|
|
{
|
|
case X: { local_offset_vec = local_offset * Vec3d::UnitX(); break; }
|
|
case Y: { local_offset_vec = local_offset * Vec3d::UnitY(); break;}
|
|
case Z: { local_offset_vec = local_offset * Vec3d::UnitZ(); break;
|
|
}
|
|
default: break;
|
|
}
|
|
if (m_object_manipulation->is_world_coordinates()) {
|
|
m_offset = local_offset_vec;
|
|
} else {//if (m_object_manipulation->is_instance_coordinates())
|
|
m_offset = m_grabbers_tran.get_matrix_no_offset() * local_offset_vec;
|
|
}
|
|
}
|
|
else
|
|
m_offset = Vec3d::Zero();
|
|
}
|
|
}
|
|
|
|
void GLGizmoScale3D::do_scale_uniform(const UpdateData& data)
|
|
{
|
|
double ratio = calc_ratio(data);
|
|
if (ratio > 0.0)
|
|
{
|
|
m_scale = m_starting.scale * ratio;
|
|
m_offset = Vec3d::Zero();
|
|
}
|
|
}
|
|
|
|
double GLGizmoScale3D::calc_ratio(const UpdateData& data) const
|
|
{
|
|
double ratio = 0.0;
|
|
|
|
Vec3d pivot = (m_starting.ctrl_down && (m_hover_id < 6)) ? m_starting.constraint_position : m_starting.plane_center; // plane_center = bottom center
|
|
Vec3d starting_vec = m_starting.drag_position - pivot;
|
|
double len_starting_vec = starting_vec.norm();
|
|
if (len_starting_vec != 0.0)
|
|
{
|
|
Vec3d mouse_dir = data.mouse_ray.unit_vector();
|
|
Vec3d plane_normal = m_starting.plane_nromal;
|
|
if (m_hover_id == 5) {
|
|
// get z-axis moving plane normal
|
|
Vec3d plane_vec = mouse_dir.cross(m_starting.plane_nromal);
|
|
plane_normal = plane_vec.cross(m_starting.plane_nromal);
|
|
}
|
|
plane_normal = plane_normal.normalized();
|
|
// finds the intersection of the mouse ray with the plane that the drag point moves
|
|
// use ray-plane intersection see i.e. https://en.wikipedia.org/wiki/Line%E2%80%93plane_intersection
|
|
auto dot_value = (plane_normal.dot(mouse_dir));
|
|
auto angle = Geometry::rad2deg(acos(dot_value));
|
|
auto big_than_min_angle = abs(angle) < 95 && abs(angle) > 85;
|
|
if (big_than_min_angle) {
|
|
return 1;
|
|
}
|
|
Vec3d inters = GetIntersectionOfRayAndPlane(data.mouse_ray.a, mouse_dir, m_starting.drag_position, plane_normal);
|
|
|
|
Vec3d inters_vec = inters - m_starting.drag_position;
|
|
|
|
// finds projection of the vector along the staring direction
|
|
double proj = inters_vec.dot(starting_vec.normalized());
|
|
|
|
ratio = (len_starting_vec + proj) / len_starting_vec;
|
|
}
|
|
|
|
if (wxGetKeyState(WXK_SHIFT))
|
|
ratio = m_snap_step * (double)std::round(ratio / m_snap_step);
|
|
|
|
return ratio;
|
|
}
|
|
|
|
} // namespace GUI
|
|
} // namespace Slic3r
|