mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-24 09:11:23 -06:00 
			
		
		
		
	Rectangle selection in 3D scene -> hovering detection
This commit is contained in:
		
							parent
							
								
									ec2f319a3d
								
							
						
					
					
						commit
						11490dfb06
					
				
					 6 changed files with 148 additions and 26 deletions
				
			
		|  | @ -223,7 +223,8 @@ void GLIndexedVertexArray::render( | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const float GLVolume::SELECTED_COLOR[4] = { 0.0f, 1.0f, 0.0f, 1.0f }; | const float GLVolume::SELECTED_COLOR[4] = { 0.0f, 1.0f, 0.0f, 1.0f }; | ||||||
| const float GLVolume::HOVER_COLOR[4] = { 0.4f, 0.9f, 0.1f, 1.0f }; | const float GLVolume::HOVER_SELECT_COLOR[4] = { 0.4f, 0.9f, 0.1f, 1.0f }; | ||||||
|  | const float GLVolume::HOVER_DESELECT_COLOR[4] = { 0.9f, 0.4f, 0.1f, 1.0f }; | ||||||
| const float GLVolume::OUTSIDE_COLOR[4] = { 0.0f, 0.38f, 0.8f, 1.0f }; | const float GLVolume::OUTSIDE_COLOR[4] = { 0.0f, 0.38f, 0.8f, 1.0f }; | ||||||
| const float GLVolume::SELECTED_OUTSIDE_COLOR[4] = { 0.19f, 0.58f, 1.0f, 1.0f }; | const float GLVolume::SELECTED_OUTSIDE_COLOR[4] = { 0.19f, 0.58f, 1.0f, 1.0f }; | ||||||
| const float GLVolume::DISABLED_COLOR[4] = { 0.25f, 0.25f, 0.25f, 1.0f }; | const float GLVolume::DISABLED_COLOR[4] = { 0.25f, 0.25f, 0.25f, 1.0f }; | ||||||
|  | @ -251,7 +252,8 @@ GLVolume::GLVolume(float r, float g, float b, float a) | ||||||
|     , zoom_to_volumes(true) |     , zoom_to_volumes(true) | ||||||
|     , shader_outside_printer_detection_enabled(false) |     , shader_outside_printer_detection_enabled(false) | ||||||
|     , is_outside(false) |     , is_outside(false) | ||||||
|     , hover(false) |     , hover_select(false) | ||||||
|  |     , hover_deselect(false) | ||||||
|     , is_modifier(false) |     , is_modifier(false) | ||||||
|     , is_wipe_tower(false) |     , is_wipe_tower(false) | ||||||
|     , is_extrusion_path(false) |     , is_extrusion_path(false) | ||||||
|  | @ -291,10 +293,12 @@ void GLVolume::set_render_color() | ||||||
|     if (force_native_color) |     if (force_native_color) | ||||||
|         set_render_color(color, 4); |         set_render_color(color, 4); | ||||||
|     else { |     else { | ||||||
|         if (selected) |         if (hover_select) | ||||||
|  |             set_render_color(HOVER_SELECT_COLOR, 4); | ||||||
|  |         else if (hover_deselect) | ||||||
|  |             set_render_color(HOVER_DESELECT_COLOR, 4); | ||||||
|  |         else if (selected) | ||||||
|             set_render_color(is_outside ? SELECTED_OUTSIDE_COLOR : SELECTED_COLOR, 4); |             set_render_color(is_outside ? SELECTED_OUTSIDE_COLOR : SELECTED_COLOR, 4); | ||||||
|         else if (hover) |  | ||||||
|             set_render_color(HOVER_COLOR, 4); |  | ||||||
|         else if (disabled) |         else if (disabled) | ||||||
|             set_render_color(DISABLED_COLOR, 4); |             set_render_color(DISABLED_COLOR, 4); | ||||||
|         else if (is_outside && shader_outside_printer_detection_enabled) |         else if (is_outside && shader_outside_printer_detection_enabled) | ||||||
|  |  | ||||||
|  | @ -225,7 +225,8 @@ private: | ||||||
| class GLVolume { | class GLVolume { | ||||||
| public: | public: | ||||||
|     static const float SELECTED_COLOR[4]; |     static const float SELECTED_COLOR[4]; | ||||||
|     static const float HOVER_COLOR[4]; |     static const float HOVER_SELECT_COLOR[4]; | ||||||
|  |     static const float HOVER_DESELECT_COLOR[4]; | ||||||
|     static const float OUTSIDE_COLOR[4]; |     static const float OUTSIDE_COLOR[4]; | ||||||
|     static const float SELECTED_OUTSIDE_COLOR[4]; |     static const float SELECTED_OUTSIDE_COLOR[4]; | ||||||
|     static const float DISABLED_COLOR[4]; |     static const float DISABLED_COLOR[4]; | ||||||
|  | @ -296,8 +297,10 @@ public: | ||||||
|     bool                shader_outside_printer_detection_enabled; |     bool                shader_outside_printer_detection_enabled; | ||||||
|     // Wheter or not this volume is outside print volume.
 |     // Wheter or not this volume is outside print volume.
 | ||||||
|     bool                is_outside; |     bool                is_outside; | ||||||
|     // Boolean: Is mouse over this object?
 |     // Boolean: Is mouse over this object to select it ?
 | ||||||
|     bool                hover; |     bool                hover_select; | ||||||
|  |     // Boolean: Is mouse over this object to deselect it ?
 | ||||||
|  |     bool                hover_deselect; | ||||||
|     // Wheter or not this volume has been generated from a modifier
 |     // Wheter or not this volume has been generated from a modifier
 | ||||||
|     bool                is_modifier; |     bool                is_modifier; | ||||||
|     // Wheter or not this volume has been generated from the wipe tower
 |     // Wheter or not this volume has been generated from the wipe tower
 | ||||||
|  |  | ||||||
|  | @ -1610,8 +1610,12 @@ void GLCanvas3D::render() | ||||||
| 
 | 
 | ||||||
|     wxGetApp().imgui()->new_frame(); |     wxGetApp().imgui()->new_frame(); | ||||||
| 
 | 
 | ||||||
|     // picking pass
 |     if (m_rectangle_selection.is_dragging()) | ||||||
|     _picking_pass(); |         // picking pass using rectangle selection
 | ||||||
|  |         _rectangular_selection_picking_pass(); | ||||||
|  |     else | ||||||
|  |         // regular picking pass
 | ||||||
|  |         _picking_pass(); | ||||||
| 
 | 
 | ||||||
|     // draw scene
 |     // draw scene
 | ||||||
|     glsafe(::glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT)); |     glsafe(::glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT)); | ||||||
|  | @ -2361,6 +2365,8 @@ void GLCanvas3D::on_key(wxKeyEvent& evt) | ||||||
|                     } |                     } | ||||||
|                     set_cursor(Standard); |                     set_cursor(Standard); | ||||||
|                 } |                 } | ||||||
|  |                 else if (keyCode == WXK_CONTROL) | ||||||
|  |                     m_dirty = true; | ||||||
|             } |             } | ||||||
|             else if (evt.GetEventType() == wxEVT_KEY_DOWN) { |             else if (evt.GetEventType() == wxEVT_KEY_DOWN) { | ||||||
|                 m_tab_down = keyCode == WXK_TAB && !evt.HasAnyModifiers(); |                 m_tab_down = keyCode == WXK_TAB && !evt.HasAnyModifiers(); | ||||||
|  | @ -2374,6 +2380,8 @@ void GLCanvas3D::on_key(wxKeyEvent& evt) | ||||||
|                     if (m_picking_enabled && (m_gizmos.get_current_type() != GLGizmosManager::SlaSupports)) |                     if (m_picking_enabled && (m_gizmos.get_current_type() != GLGizmosManager::SlaSupports)) | ||||||
|                         set_cursor(Cross); |                         set_cursor(Cross); | ||||||
|                 } |                 } | ||||||
|  |                 else if (keyCode == WXK_CONTROL) | ||||||
|  |                     m_dirty = true; | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | @ -3662,7 +3670,7 @@ void GLCanvas3D::_picking_pass() const | ||||||
|         if (inside) |         if (inside) | ||||||
|         { |         { | ||||||
|             glsafe(::glReadPixels(pos(0), cnv_size.get_height() - pos(1) - 1, 1, 1, GL_RGBA, GL_UNSIGNED_BYTE, (void*)color)); |             glsafe(::glReadPixels(pos(0), cnv_size.get_height() - pos(1) - 1, 1, 1, GL_RGBA, GL_UNSIGNED_BYTE, (void*)color)); | ||||||
|             volume_id = color[0] + color[1] * 256 + color[2] * 256 * 256; |             volume_id = color[0] + (color[1] << 8) + (color[2] << 16); | ||||||
|         } |         } | ||||||
|         if ((0 <= volume_id) && (volume_id < (int)m_volumes.volumes.size())) |         if ((0 <= volume_id) && (volume_id < (int)m_volumes.volumes.size())) | ||||||
|         { |         { | ||||||
|  | @ -3676,6 +3684,83 @@ void GLCanvas3D::_picking_pass() const | ||||||
|     } |     } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void GLCanvas3D::_rectangular_selection_picking_pass() const | ||||||
|  | { | ||||||
|  |     m_gizmos.set_hover_id(-1); | ||||||
|  | 
 | ||||||
|  |     std::set<int> idxs; | ||||||
|  | 
 | ||||||
|  |     if (m_picking_enabled) | ||||||
|  |     { | ||||||
|  |         if (m_multisample_allowed) | ||||||
|  |             glsafe(::glDisable(GL_MULTISAMPLE)); | ||||||
|  | 
 | ||||||
|  |         glsafe(::glDisable(GL_BLEND)); | ||||||
|  |         glsafe(::glEnable(GL_DEPTH_TEST)); | ||||||
|  | 
 | ||||||
|  |         glsafe(::glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT)); | ||||||
|  | 
 | ||||||
|  |         _render_volumes_for_picking(); | ||||||
|  | 
 | ||||||
|  |         if (m_multisample_allowed) | ||||||
|  |             glsafe(::glEnable(GL_MULTISAMPLE)); | ||||||
|  | 
 | ||||||
|  |         int width = (int)m_rectangle_selection.get_width(); | ||||||
|  |         int height = (int)m_rectangle_selection.get_height(); | ||||||
|  |         int px_count = width * height; | ||||||
|  | 
 | ||||||
|  |         if (px_count > 0) | ||||||
|  |         { | ||||||
|  |             int left = (int)m_rectangle_selection.get_left(); | ||||||
|  |             int top = get_canvas_size().get_height() - (int)m_rectangle_selection.get_top(); | ||||||
|  |             if ((left >= 0) && (top >= 0)) | ||||||
|  |             { | ||||||
|  | #define USE_PARALLEL 1 | ||||||
|  | #if USE_PARALLEL | ||||||
|  |                 struct Pixel | ||||||
|  |                 { | ||||||
|  |                     std::array<GLubyte, 4> data; | ||||||
|  |                     int id() const { return data[0] + (data[1] << 8) + (data[2] << 16); } | ||||||
|  |                 }; | ||||||
|  | 
 | ||||||
|  |                 std::vector<Pixel> frame(px_count); | ||||||
|  |                 glsafe(::glReadPixels(left, top, width, height, GL_RGBA, GL_UNSIGNED_BYTE, (void*)frame.data())); | ||||||
|  | 
 | ||||||
|  |                 tbb::spin_mutex mutex; | ||||||
|  |                 tbb::parallel_for(tbb::blocked_range<size_t>(0, frame.size(), (size_t)width), | ||||||
|  |                     [this, &frame, &idxs, &mutex](const tbb::blocked_range<size_t>& range) { | ||||||
|  |                     for (size_t i = range.begin(); i < range.end(); ++i) | ||||||
|  |                     { | ||||||
|  |                         int volume_id = frame[i].id(); | ||||||
|  |                         if ((0 <= volume_id) && (volume_id < (int)m_volumes.volumes.size())) | ||||||
|  |                         { | ||||||
|  |                             mutex.lock(); | ||||||
|  |                             idxs.insert(volume_id); | ||||||
|  |                             mutex.unlock(); | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |                 ); | ||||||
|  | #else | ||||||
|  |                 std::vector<GLubyte> frame(4 * px_count); | ||||||
|  |                 glsafe(::glReadPixels(left, top, width, height, GL_RGBA, GL_UNSIGNED_BYTE, (void*)frame.data())); | ||||||
|  | 
 | ||||||
|  |                 for (int i = 0; i < px_count; ++i) | ||||||
|  |                 { | ||||||
|  |                     int px_id = 4 * i; | ||||||
|  |                     int volume_id = frame[px_id] + (frame[px_id + 1] << 8) + (frame[px_id + 2] << 16); | ||||||
|  |                     if ((0 <= volume_id) && (volume_id < (int)m_volumes.volumes.size())) | ||||||
|  |                         idxs.insert(volume_id); | ||||||
|  |                 } | ||||||
|  | #endif // USE_PARALLEL
 | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     m_hover_volume_idxs.assign(idxs.begin(), idxs.end()); | ||||||
|  |     _update_volumes_hover_state(); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| void GLCanvas3D::_render_background() const | void GLCanvas3D::_render_background() const | ||||||
| { | { | ||||||
|     glsafe(::glPushMatrix()); |     glsafe(::glPushMatrix()); | ||||||
|  | @ -4161,24 +4246,48 @@ void GLCanvas3D::_update_volumes_hover_state() const | ||||||
| { | { | ||||||
|     for (GLVolume* v : m_volumes.volumes) |     for (GLVolume* v : m_volumes.volumes) | ||||||
|     { |     { | ||||||
|         v->hover = false; |         v->hover_select = false; | ||||||
|  |         v->hover_deselect = false; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (m_hover_volume_idxs.empty()) |     if (m_hover_volume_idxs.empty()) | ||||||
|         return; |         return; | ||||||
| 
 | 
 | ||||||
|     GLVolume* volume = m_volumes.volumes[get_first_hover_volume_idx()]; |     bool is_ctrl_pressed = wxGetKeyState(WXK_CONTROL); | ||||||
|     if (volume->is_modifier) |     bool is_shift_pressed = wxGetKeyState(WXK_SHIFT); | ||||||
|         volume->hover = true; |     bool is_alt_pressed = wxGetKeyState(WXK_ALT); | ||||||
|     else |  | ||||||
|     { |  | ||||||
|         int object_idx = volume->object_idx(); |  | ||||||
|         int instance_idx = volume->instance_idx(); |  | ||||||
| 
 | 
 | ||||||
|         for (GLVolume* v : m_volumes.volumes) |     for (int i : m_hover_volume_idxs) | ||||||
|  |     { | ||||||
|  |         GLVolume* volume = m_volumes.volumes[i]; | ||||||
|  |         bool deselect = volume->selected && ((is_ctrl_pressed && !is_shift_pressed) || (!is_ctrl_pressed && (m_rectangle_selection.get_state() == GLSelectionRectangle::Deselect))); | ||||||
|  |         bool select = !volume->selected || (volume->is_modifier && ((is_ctrl_pressed && !is_alt_pressed) || (!is_ctrl_pressed && (!m_rectangle_selection.is_dragging() || (m_rectangle_selection.get_state() == GLSelectionRectangle::Select))))); | ||||||
|  | 
 | ||||||
|  |         if (select || deselect) | ||||||
|         { |         { | ||||||
|             if ((v->object_idx() == object_idx) && (v->instance_idx() == instance_idx)) |             if (volume->is_modifier && (!deselect || ((volume->object_idx() == m_selection.get_object_idx()) && (volume->instance_idx() == m_selection.get_instance_idx())))) | ||||||
|                 v->hover = true; |             { | ||||||
|  |                 if (deselect) | ||||||
|  |                     volume->hover_deselect = true; | ||||||
|  |                 else | ||||||
|  |                     volume->hover_select = true; | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 int object_idx = volume->object_idx(); | ||||||
|  |                 int instance_idx = volume->instance_idx(); | ||||||
|  | 
 | ||||||
|  |                 for (GLVolume* v : m_volumes.volumes) | ||||||
|  |                 { | ||||||
|  |                     if ((v->object_idx() == object_idx) && (v->instance_idx() == instance_idx)) | ||||||
|  |                     { | ||||||
|  |                         if (deselect) | ||||||
|  |                             v->hover_deselect = true; | ||||||
|  |                         else | ||||||
|  |                             v->hover_select = true; | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -623,6 +623,7 @@ private: | ||||||
|     void _refresh_if_shown_on_screen(); |     void _refresh_if_shown_on_screen(); | ||||||
| 
 | 
 | ||||||
|     void _picking_pass() const; |     void _picking_pass() const; | ||||||
|  |     void _rectangular_selection_picking_pass() const; | ||||||
|     void _render_background() const; |     void _render_background() const; | ||||||
|     void _render_bed(float theta) const; |     void _render_bed(float theta) const; | ||||||
|     void _render_axes() const; |     void _render_axes() const; | ||||||
|  |  | ||||||
|  | @ -59,10 +59,8 @@ namespace GUI { | ||||||
| 
 | 
 | ||||||
|     void GLSelectionRectangle::stop_dragging() |     void GLSelectionRectangle::stop_dragging() | ||||||
|     { |     { | ||||||
|         if (!is_dragging()) |         if (is_dragging()) | ||||||
|             return; |             m_state = Off; | ||||||
| 
 |  | ||||||
|         m_state = Off; |  | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void GLSelectionRectangle::render(const GLCanvas3D& canvas) const |     void GLSelectionRectangle::render(const GLCanvas3D& canvas) const | ||||||
|  |  | ||||||
|  | @ -35,6 +35,13 @@ public: | ||||||
|     bool is_dragging() const { return m_state != Off; } |     bool is_dragging() const { return m_state != Off; } | ||||||
|     EState get_state() const { return m_state; } |     EState get_state() const { return m_state; } | ||||||
|      |      | ||||||
|  |     float get_width() const { return std::abs(m_start_corner(0) - m_end_corner(0)); } | ||||||
|  |     float get_height() const { return std::abs(m_start_corner(1) - m_end_corner(1)); } | ||||||
|  |     float get_left() const { return std::min(m_start_corner(0), m_end_corner(0)); } | ||||||
|  |     float get_right() const { return std::max(m_start_corner(0), m_end_corner(0)); } | ||||||
|  |     float get_top() const { return std::max(m_start_corner(1), m_end_corner(1)); } | ||||||
|  |     float get_bottom() const { return std::min(m_start_corner(1), m_end_corner(1)); } | ||||||
|  | 
 | ||||||
| private: | private: | ||||||
|     EState m_state = Off; |     EState m_state = Off; | ||||||
|     Vec2d m_start_corner; |     Vec2d m_start_corner; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Enrico Turri
						Enrico Turri