mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-26 02:01:12 -06:00 
			
		
		
		
	Added seed fill for MMU segmentation
This commit is contained in:
		
							parent
							
								
									be1b4ce18c
								
							
						
					
					
						commit
						576c5b78e9
					
				
					 8 changed files with 228 additions and 174 deletions
				
			
		|  | @ -13,20 +13,15 @@ | |||
| #include <GL/glew.h> | ||||
| 
 | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| namespace GUI { | ||||
| 
 | ||||
| 
 | ||||
| namespace Slic3r::GUI { | ||||
| 
 | ||||
| void GLGizmoMmuSegmentation::on_shutdown() | ||||
| { | ||||
|     m_angle_threshold_deg = 0.f; | ||||
| //    m_seed_fill_angle = 0.f;
 | ||||
| //    m_seed_fill_enabled = false;
 | ||||
|     m_parent.use_slope(false); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| std::string GLGizmoMmuSegmentation::on_get_name() const | ||||
| { | ||||
|     // FIXME Lukas H.: Discuss and change shortcut
 | ||||
|  | @ -44,28 +39,24 @@ bool GLGizmoMmuSegmentation::on_init() | |||
|     // FIXME Lukas H.: Discuss and change shortcut
 | ||||
|     m_shortcut_key = WXK_CONTROL_N; | ||||
| 
 | ||||
|     m_desc["clipping_of_view"] = _L("Clipping of view") + ": "; | ||||
|     m_desc["reset_direction"]  = _L("Reset direction"); | ||||
|     m_desc["cursor_size"]      = _L("Brush size") + ": "; | ||||
|     m_desc["cursor_type"]      = _L("Brush shape") + ": "; | ||||
|     m_desc["enforce_caption"]  = _L("Left mouse button") + ": "; | ||||
|     m_desc["enforce"]          = _L("Enforce supports"); | ||||
|     m_desc["block_caption"]    = _L("Right mouse button") + ": "; | ||||
|     m_desc["block"]            = _L("Block supports"); | ||||
|     m_desc["remove_caption"]   = _L("Shift + Left mouse button") + ": "; | ||||
|     m_desc["remove"]           = _L("Remove selection"); | ||||
|     m_desc["remove_all"]       = _L("Remove all selection"); | ||||
|     m_desc["circle"]           = _L("Circle"); | ||||
|     m_desc["sphere"]           = _L("Sphere"); | ||||
|     m_desc["highlight_by_angle"] = _L("Highlight by angle"); | ||||
|     m_desc["enforce_button"]   = _L("Enforce"); | ||||
|     m_desc["cancel"]           = _L("Cancel"); | ||||
|     m_desc["reset_direction"]      = _L("Reset direction"); | ||||
|     m_desc["clipping_of_view"]     = _L("Clipping of view") + ": "; | ||||
|     m_desc["cursor_size"]          = _L("Brush size") + ": "; | ||||
|     m_desc["cursor_type"]          = _L("Brush shape") + ": "; | ||||
|     m_desc["first_color_caption"]  = _L("Left mouse button") + ": "; | ||||
|     m_desc["first_color"]          = _L("First color"); | ||||
|     m_desc["second_color_caption"] = _L("Right mouse button") + ": "; | ||||
|     m_desc["second_color"]         = _L("Second color"); | ||||
|     m_desc["remove_caption"]       = _L("Shift + Left mouse button") + ": "; | ||||
|     m_desc["remove"]               = _L("Remove painted color"); | ||||
|     m_desc["remove_all"]           = _L("Remove all painted colors"); | ||||
|     m_desc["circle"]               = _L("Circle"); | ||||
|     m_desc["sphere"]               = _L("Sphere"); | ||||
|     m_desc["seed_fill_angle"]      = _L("Seed fill angle"); | ||||
| 
 | ||||
|     return true; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| void GLGizmoMmuSegmentation::render_painter_gizmo() const | ||||
| { | ||||
|     const Selection& selection = m_parent.get_selection(); | ||||
|  | @ -81,99 +72,81 @@ void GLGizmoMmuSegmentation::render_painter_gizmo() const | |||
|     glsafe(::glDisable(GL_BLEND)); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| void GLGizmoMmuSegmentation::on_render_input_window(float x, float y, float bottom_limit) | ||||
| { | ||||
|     if (! m_c->selection_info()->model_object()) | ||||
|     if (!m_c->selection_info()->model_object()) | ||||
|         return; | ||||
| 
 | ||||
|     const float approx_height = m_imgui->scaled(17.0f); | ||||
|     y = std::min(y, bottom_limit - approx_height); | ||||
|     const float approx_height = m_imgui->scaled(23.0f); | ||||
|                             y = std::min(y, bottom_limit - approx_height); | ||||
|     m_imgui->set_next_window_pos(x, y, ImGuiCond_Always); | ||||
| 
 | ||||
|     m_imgui->begin(on_get_name(), ImGuiWindowFlags_NoMove | ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoCollapse); | ||||
| 
 | ||||
|     // First calculate width of all the texts that are could possibly be shown. We will decide set the dialog width based on that:
 | ||||
|     const float clipping_slider_left = std::max(m_imgui->calc_text_size(m_desc.at("clipping_of_view")).x, | ||||
|                                                 m_imgui->calc_text_size(m_desc.at("reset_direction")).x) | ||||
|                                           + m_imgui->scaled(1.5f); | ||||
|     const float cursor_slider_left = m_imgui->calc_text_size(m_desc.at("cursor_size")).x + m_imgui->scaled(1.f); | ||||
|     const float autoset_slider_left = m_imgui->calc_text_size(m_desc.at("highlight_by_angle")).x + m_imgui->scaled(1.f); | ||||
|     const float cursor_type_radio_left  = m_imgui->calc_text_size(m_desc.at("cursor_type")).x + m_imgui->scaled(1.f); | ||||
|     const float cursor_type_radio_width1 = m_imgui->calc_text_size(m_desc["circle"]).x | ||||
|                                              + m_imgui->scaled(2.5f); | ||||
|     const float cursor_type_radio_width2 = m_imgui->calc_text_size(m_desc["sphere"]).x | ||||
|                                              + m_imgui->scaled(2.5f); | ||||
|     const float button_width = m_imgui->calc_text_size(m_desc.at("remove_all")).x + m_imgui->scaled(1.f); | ||||
|     const float button_enforce_width = m_imgui->calc_text_size(m_desc.at("enforce_button")).x; | ||||
|     const float button_cancel_width = m_imgui->calc_text_size(m_desc.at("cancel")).x; | ||||
|     const float buttons_width = std::max(button_enforce_width, button_cancel_width) + m_imgui->scaled(0.5f); | ||||
|     const float minimal_slider_width = m_imgui->scaled(4.f); | ||||
|                                                 m_imgui->calc_text_size(m_desc.at("reset_direction")).x) + m_imgui->scaled(1.5f); | ||||
|     const float cursor_slider_left       = m_imgui->calc_text_size(m_desc.at("cursor_size")).x + m_imgui->scaled(1.f); | ||||
|     const float autoset_slider_left      = m_imgui->calc_text_size(m_desc.at("seed_fill_angle")).x + m_imgui->scaled(1.f); | ||||
|     const float cursor_type_radio_left   = m_imgui->calc_text_size(m_desc.at("cursor_type")).x + m_imgui->scaled(1.f); | ||||
|     const float cursor_type_radio_width1 = m_imgui->calc_text_size(m_desc["circle"]).x + m_imgui->scaled(2.5f); | ||||
|     const float cursor_type_radio_width2 = m_imgui->calc_text_size(m_desc["sphere"]).x + m_imgui->scaled(2.5f); | ||||
|     const float button_width             = m_imgui->calc_text_size(m_desc.at("remove_all")).x + m_imgui->scaled(1.f); | ||||
|     const float buttons_width            = m_imgui->scaled(0.5f); | ||||
|     const float minimal_slider_width     = m_imgui->scaled(4.f); | ||||
| 
 | ||||
|     float caption_max = 0.f; | ||||
|     float caption_max    = 0.f; | ||||
|     float total_text_max = 0.; | ||||
|     for (const std::string& t : {"enforce", "block", "remove"}) { | ||||
|         caption_max = std::max(caption_max, m_imgui->calc_text_size(m_desc.at(t+"_caption")).x); | ||||
|     for (const std::string &t : {"first_color", "second_color", "remove"}) { | ||||
|         caption_max    = std::max(caption_max, m_imgui->calc_text_size(m_desc.at(t + "_caption")).x); | ||||
|         total_text_max = std::max(total_text_max, caption_max + m_imgui->calc_text_size(m_desc.at(t)).x); | ||||
|     } | ||||
|     caption_max += m_imgui->scaled(1.f); | ||||
|     total_text_max += m_imgui->scaled(1.f); | ||||
| 
 | ||||
|     float window_width = minimal_slider_width + std::max(autoset_slider_left, std::max(cursor_slider_left, clipping_slider_left)); | ||||
|     window_width = std::max(window_width, total_text_max); | ||||
|     window_width = std::max(window_width, button_width); | ||||
|     window_width = std::max(window_width, cursor_type_radio_left + cursor_type_radio_width1 + cursor_type_radio_width2); | ||||
|     window_width = std::max(window_width, 2.f * buttons_width + m_imgui->scaled(1.f)); | ||||
|     window_width       = std::max(window_width, total_text_max); | ||||
|     window_width       = std::max(window_width, button_width); | ||||
|     window_width       = std::max(window_width, cursor_type_radio_left + cursor_type_radio_width1 + cursor_type_radio_width2); | ||||
|     window_width       = std::max(window_width, 2.f * buttons_width + m_imgui->scaled(1.f)); | ||||
| 
 | ||||
|     auto draw_text_with_caption = [this, &caption_max](const wxString& caption, const wxString& text) { | ||||
|     auto draw_text_with_caption = [this, &caption_max](const wxString &caption, const wxString &text) { | ||||
|         m_imgui->text_colored(ImGuiWrapper::COL_ORANGE_LIGHT, caption); | ||||
|         ImGui::SameLine(caption_max); | ||||
|         m_imgui->text(text); | ||||
|     }; | ||||
| 
 | ||||
|     for (const std::string& t : {"enforce", "block", "remove"}) | ||||
|     for (const std::string &t : {"first_color", "second_color", "remove"}) | ||||
|         draw_text_with_caption(m_desc.at(t + "_caption"), m_desc.at(t)); | ||||
| 
 | ||||
|     m_imgui->text(""); | ||||
|     ImGui::Separator(); | ||||
| 
 | ||||
|     m_imgui->text(m_desc["highlight_by_angle"] + ":"); | ||||
|     if (m_imgui->checkbox(_L("Seed fill"), m_seed_fill_enabled)) | ||||
|         if (!m_seed_fill_enabled) | ||||
|             for (auto &triangle_selector : m_triangle_selectors) | ||||
|                 triangle_selector->seed_fill_unselect_all_triangles(); | ||||
| 
 | ||||
|     m_imgui->text(m_desc["seed_fill_angle"] + ":"); | ||||
|     ImGui::AlignTextToFramePadding(); | ||||
|     std::string format_str = std::string("%.f") + I18N::translate_utf8("°", | ||||
|         "Degree sign to use in the respective slider in FDM supports gizmo," | ||||
|         "placed after the number with no whitespace in between."); | ||||
|     std::string format_str = std::string("%.f") + I18N::translate_utf8("°", "Degree sign to use in the respective slider in FDM supports gizmo," | ||||
|                                                                             "placed after the number with no whitespace in between."); | ||||
|     ImGui::SameLine(autoset_slider_left); | ||||
|     ImGui::PushItemWidth(window_width - autoset_slider_left); | ||||
|     if (m_imgui->slider_float("", &m_angle_threshold_deg, 0.f, 90.f, format_str.data())) { | ||||
|         m_parent.set_slope_normal_angle(90.f - m_angle_threshold_deg); | ||||
|         if (! m_parent.is_using_slope()) { | ||||
|             m_parent.use_slope(true); | ||||
|             m_parent.set_as_dirty(); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     m_imgui->disabled_begin(m_angle_threshold_deg == 0.f); | ||||
|     ImGui::NewLine(); | ||||
|     ImGui::SameLine(window_width - 2.f*buttons_width - m_imgui->scaled(0.5f)); | ||||
|     if (m_imgui->button(m_desc["enforce_button"], buttons_width, 0.f)) { | ||||
|         select_facets_by_angle(m_angle_threshold_deg, false); | ||||
|         m_angle_threshold_deg = 0.f; | ||||
|     } | ||||
|     ImGui::SameLine(window_width - buttons_width); | ||||
|     if (m_imgui->button(m_desc["cancel"], buttons_width, 0.f)) { | ||||
|         m_angle_threshold_deg = 0.f; | ||||
|         m_parent.use_slope(false); | ||||
|     } | ||||
|     m_imgui->disabled_begin(!m_seed_fill_enabled); | ||||
|     m_imgui->slider_float("", &m_seed_fill_angle, 0.f, 90.f, format_str.data()); | ||||
|     m_imgui->disabled_end(); | ||||
| 
 | ||||
|     ImGui::NewLine(); | ||||
|     ImGui::SameLine(window_width - 2.f * buttons_width - m_imgui->scaled(0.5f)); | ||||
| 
 | ||||
|     ImGui::Separator(); | ||||
| 
 | ||||
|     if (m_imgui->button(m_desc.at("remove_all"))) { | ||||
|         Plater::TakeSnapshot(wxGetApp().plater(), wxString(_L("Reset selection"))); | ||||
|         ModelObject* mo = m_c->selection_info()->model_object(); | ||||
|         int idx = -1; | ||||
|         for (ModelVolume* mv : mo->volumes) { | ||||
|         ModelObject *mo  = m_c->selection_info()->model_object(); | ||||
|         int          idx = -1; | ||||
|         for (ModelVolume *mv : mo->volumes) { | ||||
|             if (mv->is_model_part()) { | ||||
|                 ++idx; | ||||
|                 m_triangle_selectors[idx]->reset(); | ||||
|  | @ -184,7 +157,6 @@ void GLGizmoMmuSegmentation::on_render_input_window(float x, float y, float bott | |||
|         m_parent.set_as_dirty(); | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     const float max_tooltip_width = ImGui::GetFontSize() * 20.0f; | ||||
| 
 | ||||
|     ImGui::AlignTextToFramePadding(); | ||||
|  | @ -200,7 +172,6 @@ void GLGizmoMmuSegmentation::on_render_input_window(float x, float y, float bott | |||
|         ImGui::EndTooltip(); | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     ImGui::AlignTextToFramePadding(); | ||||
|     m_imgui->text(m_desc.at("cursor_type")); | ||||
|     ImGui::SameLine(cursor_type_radio_left + m_imgui->scaled(0.f)); | ||||
|  | @ -221,7 +192,7 @@ void GLGizmoMmuSegmentation::on_render_input_window(float x, float y, float bott | |||
|     ImGui::SameLine(cursor_type_radio_left + cursor_type_radio_width2 + m_imgui->scaled(0.f)); | ||||
|     ImGui::PushItemWidth(cursor_type_radio_width2); | ||||
| 
 | ||||
|     if (m_imgui->radio_button(m_desc["circle"], ! sphere_sel)) | ||||
|     if (m_imgui->radio_button(m_desc["circle"], !sphere_sel)) | ||||
|         sphere_sel = false; | ||||
| 
 | ||||
|     if (ImGui::IsItemHovered()) { | ||||
|  | @ -232,23 +203,17 @@ void GLGizmoMmuSegmentation::on_render_input_window(float x, float y, float bott | |||
|         ImGui::EndTooltip(); | ||||
|     } | ||||
| 
 | ||||
|     m_cursor_type = sphere_sel | ||||
|             ? TriangleSelector::CursorType::SPHERE | ||||
|             : TriangleSelector::CursorType::CIRCLE; | ||||
| 
 | ||||
| 
 | ||||
|     m_cursor_type = sphere_sel ? TriangleSelector::CursorType::SPHERE : TriangleSelector::CursorType::CIRCLE; | ||||
| 
 | ||||
|     m_imgui->checkbox(_L("Split triangles"), m_triangle_splitting_enabled); | ||||
| 
 | ||||
|     ImGui::Separator(); | ||||
|     if (m_c->object_clipper()->get_position() == 0.f) { | ||||
|         ImGui::AlignTextToFramePadding(); | ||||
|         m_imgui->text(m_desc.at("clipping_of_view")); | ||||
|     } | ||||
|     else { | ||||
|     } else { | ||||
|         if (m_imgui->button(m_desc.at("reset_direction"))) { | ||||
|             wxGetApp().CallAfter([this](){ | ||||
|                     m_c->object_clipper()->set_position(-1., false); | ||||
|                 }); | ||||
|             wxGetApp().CallAfter([this]() { m_c->object_clipper()->set_position(-1., false); }); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|  | @ -256,7 +221,7 @@ void GLGizmoMmuSegmentation::on_render_input_window(float x, float y, float bott | |||
|     ImGui::PushItemWidth(window_width - clipping_slider_left); | ||||
|     float clp_dist = m_c->object_clipper()->get_position(); | ||||
|     if (ImGui::SliderFloat("  ", &clp_dist, 0.f, 1.f, "%.2f")) | ||||
|     m_c->object_clipper()->set_position(clp_dist, true); | ||||
|         m_c->object_clipper()->set_position(clp_dist, true); | ||||
|     if (ImGui::IsItemHovered()) { | ||||
|         ImGui::BeginTooltip(); | ||||
|         ImGui::PushTextWrapPos(max_tooltip_width); | ||||
|  | @ -267,50 +232,6 @@ void GLGizmoMmuSegmentation::on_render_input_window(float x, float y, float bott | |||
|     m_imgui->end(); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| void GLGizmoMmuSegmentation::select_facets_by_angle(float threshold_deg, bool block) | ||||
| { | ||||
|     float threshold = (M_PI/180.)*threshold_deg; | ||||
|     const Selection& selection = m_parent.get_selection(); | ||||
|     const ModelObject* mo = m_c->selection_info()->model_object(); | ||||
|     const ModelInstance* mi = mo->instances[selection.get_instance_idx()]; | ||||
| 
 | ||||
|     int mesh_id = -1; | ||||
|     for (const ModelVolume* mv : mo->volumes) { | ||||
|         if (! mv->is_model_part()) | ||||
|             continue; | ||||
| 
 | ||||
|         ++mesh_id; | ||||
| 
 | ||||
|         const Transform3d trafo_matrix = mi->get_matrix(true) * mv->get_matrix(true); | ||||
|         Vec3f down  = (trafo_matrix.inverse() * (-Vec3d::UnitZ())).cast<float>().normalized(); | ||||
|         Vec3f limit = (trafo_matrix.inverse() * Vec3d(std::sin(threshold), 0, -std::cos(threshold))).cast<float>().normalized(); | ||||
| 
 | ||||
|         float dot_limit = limit.dot(down); | ||||
| 
 | ||||
|         // Now calculate dot product of vert_direction and facets' normals.
 | ||||
|         int idx = -1; | ||||
|         for (const stl_facet& facet : mv->mesh().stl.facet_start) { | ||||
|             ++idx; | ||||
|             if (facet.normal.dot(down) > dot_limit) | ||||
|                 m_triangle_selectors[mesh_id]->set_facet(idx, | ||||
|                                                          block | ||||
|                                                          ? EnforcerBlockerType::BLOCKER | ||||
|                                                          : EnforcerBlockerType::ENFORCER); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     activate_internal_undo_redo_stack(true); | ||||
| 
 | ||||
|     Plater::TakeSnapshot(wxGetApp().plater(), block ? _L("Block supports by angle") | ||||
|                                                     : _L("Add supports by angle")); | ||||
|     update_model_object(); | ||||
|     m_parent.set_as_dirty(); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| void GLGizmoMmuSegmentation::update_model_object() const | ||||
| { | ||||
|     bool updated = false; | ||||
|  | @ -327,8 +248,6 @@ void GLGizmoMmuSegmentation::update_model_object() const | |||
|         m_parent.post_event(SimpleEvent(EVT_GLCANVAS_SCHEDULE_BACKGROUND_PROCESS)); | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| void GLGizmoMmuSegmentation::update_from_model_object() | ||||
| { | ||||
|     wxBusyCursor wait; | ||||
|  | @ -351,13 +270,10 @@ void GLGizmoMmuSegmentation::update_from_model_object() | |||
|     } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| PainterGizmoType GLGizmoMmuSegmentation::get_painter_type() const | ||||
| { | ||||
|     return PainterGizmoType::MMU_SEGMENTATION; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| } // namespace GUI
 | ||||
| } // namespace Slic3r
 | ||||
|  |  | |||
|  | @ -31,9 +31,6 @@ private: | |||
|     void on_shutdown() override; | ||||
|     PainterGizmoType get_painter_type() const override; | ||||
| 
 | ||||
|     void select_facets_by_angle(float threshold, bool block); | ||||
|     float m_angle_threshold_deg = 0.f; | ||||
| 
 | ||||
|     // This map holds all translated description texts, so they can be easily referenced during layout calculations
 | ||||
|     // etc. When language changes, GUI is recreated and this class constructed again, so the change takes effect.
 | ||||
|     std::map<std::string, wxString> m_desc; | ||||
|  |  | |||
|  | @ -143,11 +143,12 @@ void GLGizmoPainterBase::render_cursor() const | |||
|     if (m_rr.mesh_id == -1) | ||||
|         return; | ||||
| 
 | ||||
| 
 | ||||
|     if (m_cursor_type == TriangleSelector::SPHERE) | ||||
|         render_cursor_sphere(trafo_matrices[m_rr.mesh_id]); | ||||
|     else | ||||
|         render_cursor_circle(); | ||||
|     if (!m_seed_fill_enabled) { | ||||
|         if (m_cursor_type == TriangleSelector::SPHERE) | ||||
|             render_cursor_sphere(trafo_matrices[m_rr.mesh_id]); | ||||
|         else | ||||
|             render_cursor_circle(); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
|  | @ -351,14 +352,50 @@ bool GLGizmoPainterBase::gizmo_event(SLAGizmoEventType action, const Vec2d& mous | |||
|             Vec3f camera_pos = (trafo_matrix.inverse() * camera.get_position()).cast<float>(); | ||||
| 
 | ||||
|             assert(m_rr.mesh_id < int(m_triangle_selectors.size())); | ||||
|             m_triangle_selectors[m_rr.mesh_id]->select_patch(m_rr.hit, m_rr.facet, camera_pos, | ||||
|                                        m_cursor_radius, m_cursor_type, new_state, trafo_matrix); | ||||
|             if (m_seed_fill_enabled) | ||||
|                 m_triangle_selectors[m_rr.mesh_id]->seed_fill_apply_on_triangles(new_state); | ||||
|             else | ||||
|                 m_triangle_selectors[m_rr.mesh_id]->select_patch(m_rr.hit, m_rr.facet, camera_pos, m_cursor_radius, m_cursor_type, | ||||
|                                                                  new_state, trafo_matrix, m_triangle_splitting_enabled); | ||||
|             m_last_mouse_click = mouse_position; | ||||
|         } | ||||
| 
 | ||||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     if (action == SLAGizmoEventType::Moving && m_seed_fill_enabled) { | ||||
|         if (m_triangle_selectors.empty()) | ||||
|             return false; | ||||
| 
 | ||||
|         const Camera &       camera         = wxGetApp().plater()->get_camera(); | ||||
|         const Selection &    selection      = m_parent.get_selection(); | ||||
|         const ModelObject *  mo             = m_c->selection_info()->model_object(); | ||||
|         const ModelInstance *mi             = mo->instances[selection.get_instance_idx()]; | ||||
|         const Transform3d &  instance_trafo = mi->get_transformation().get_matrix(); | ||||
| 
 | ||||
|         // Precalculate transformations of individual meshes.
 | ||||
|         std::vector<Transform3d> trafo_matrices; | ||||
|         for (const ModelVolume *mv : mo->volumes) | ||||
|             if (mv->is_model_part()) | ||||
|                 trafo_matrices.emplace_back(instance_trafo * mv->get_matrix()); | ||||
| 
 | ||||
|         // Now "click" into all the prepared points and spill paint around them.
 | ||||
|         update_raycast_cache(mouse_position, camera, trafo_matrices); | ||||
| 
 | ||||
|         if (m_rr.mesh_id == -1) { | ||||
|             // Clean selected by seed fill for all triangles
 | ||||
|             for (auto &triangle_selector : m_triangle_selectors) | ||||
|                 triangle_selector->seed_fill_unselect_all_triangles(); | ||||
| 
 | ||||
|             // In case we have no valid hit, we can return.
 | ||||
|             return false; | ||||
|         } | ||||
| 
 | ||||
|         assert(m_rr.mesh_id < int(m_triangle_selectors.size())); | ||||
|         m_triangle_selectors[m_rr.mesh_id]->seed_fill_select_triangles(m_rr.hit, m_rr.facet, m_seed_fill_angle); | ||||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     if ((action == SLAGizmoEventType::LeftUp || action == SLAGizmoEventType::RightUp) | ||||
|       && m_button_down != Button::None) { | ||||
|         // Take snapshot and update ModelVolume data.
 | ||||
|  | @ -521,12 +558,14 @@ void TriangleSelectorGUI::render(ImGuiWrapper* imgui) | |||
| { | ||||
|     int enf_cnt = 0; | ||||
|     int blc_cnt = 0; | ||||
|     int seed_fill_cnt = 0; | ||||
| 
 | ||||
|     m_iva_enforcers.release_geometry(); | ||||
|     m_iva_blockers.release_geometry(); | ||||
|     m_iva_seed_fill.release_geometry(); | ||||
| 
 | ||||
|     for (const Triangle& tr : m_triangles) { | ||||
|         if (! tr.valid || tr.is_split() || tr.get_state() == EnforcerBlockerType::NONE) | ||||
|         if (!tr.valid || tr.is_split() || tr.get_state() == EnforcerBlockerType::NONE || tr.is_selected_by_seed_fill()) | ||||
|             continue; | ||||
| 
 | ||||
|         GLIndexedVertexArray& va = tr.get_state() == EnforcerBlockerType::ENFORCER | ||||
|  | @ -543,17 +582,32 @@ void TriangleSelectorGUI::render(ImGuiWrapper* imgui) | |||
|                              double(tr.normal[0]), | ||||
|                              double(tr.normal[1]), | ||||
|                              double(tr.normal[2])); | ||||
|         va.push_triangle(cnt, | ||||
|                          cnt+1, | ||||
|                          cnt+2); | ||||
|         va.push_triangle(cnt, cnt + 1, cnt + 2); | ||||
|         cnt += 3; | ||||
|     } | ||||
| 
 | ||||
|     for (const Triangle &tr : m_triangles) { | ||||
|         if (!tr.valid || tr.is_split() || !tr.is_selected_by_seed_fill()) | ||||
|             continue; | ||||
| 
 | ||||
|         for (int i = 0; i < 3; ++i) | ||||
|             m_iva_seed_fill.push_geometry(double(m_vertices[tr.verts_idxs[i]].v[0]), | ||||
|                                           double(m_vertices[tr.verts_idxs[i]].v[1]), | ||||
|                                           double(m_vertices[tr.verts_idxs[i]].v[2]), | ||||
|                                           double(tr.normal[0]), | ||||
|                                           double(tr.normal[1]), | ||||
|                                           double(tr.normal[2])); | ||||
|         m_iva_seed_fill.push_triangle(seed_fill_cnt, seed_fill_cnt + 1, seed_fill_cnt + 2); | ||||
|         seed_fill_cnt += 3; | ||||
|     } | ||||
| 
 | ||||
|     m_iva_enforcers.finalize_geometry(true); | ||||
|     m_iva_blockers.finalize_geometry(true); | ||||
|     m_iva_seed_fill.finalize_geometry(true); | ||||
| 
 | ||||
|     bool render_enf = m_iva_enforcers.has_VBOs(); | ||||
|     bool render_blc = m_iva_blockers.has_VBOs(); | ||||
|     bool render_seed_fill = m_iva_seed_fill.has_VBOs(); | ||||
| 
 | ||||
|     auto* shader = wxGetApp().get_shader("gouraud"); | ||||
|     if (! shader) | ||||
|  | @ -575,6 +629,12 @@ void TriangleSelectorGUI::render(ImGuiWrapper* imgui) | |||
|         m_iva_blockers.render(); | ||||
|     } | ||||
| 
 | ||||
|     if (render_seed_fill) { | ||||
|         std::array<float, 4> color = { 0.f, 1.00f, 0.44f, 1.f }; | ||||
|         shader->set_uniform("uniform_color", color); | ||||
|         m_iva_seed_fill.render(); | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
| #ifdef PRUSASLICER_TRIANGLE_SELECTOR_DEBUG | ||||
|     if (imgui) | ||||
|  |  | |||
|  | @ -48,6 +48,7 @@ public: | |||
| private: | ||||
|     GLIndexedVertexArray m_iva_enforcers; | ||||
|     GLIndexedVertexArray m_iva_blockers; | ||||
|     GLIndexedVertexArray m_iva_seed_fill; | ||||
|     std::array<GLIndexedVertexArray, 3> m_varrays; | ||||
| }; | ||||
| 
 | ||||
|  | @ -95,6 +96,9 @@ protected: | |||
| 
 | ||||
|     TriangleSelector::CursorType m_cursor_type = TriangleSelector::SPHERE; | ||||
| 
 | ||||
|     bool  m_triangle_splitting_enabled = true; | ||||
|     bool  m_seed_fill_enabled          = false; | ||||
|     float m_seed_fill_angle            = 0.f; | ||||
| 
 | ||||
| private: | ||||
|     bool is_mesh_point_clipped(const Vec3d& point, const Transform3d& trafo) const; | ||||
|  |  | |||
|  | @ -32,7 +32,8 @@ enum class SLAGizmoEventType : unsigned char { | |||
|     ManualEditing, | ||||
|     MouseWheelUp, | ||||
|     MouseWheelDown, | ||||
|     ResetClippingPlane | ||||
|     ResetClippingPlane, | ||||
|     Moving | ||||
| }; | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
|  | @ -522,9 +522,11 @@ bool GLGizmosManager::on_mouse(wxMouseEvent& evt) | |||
|     bool control_down = evt.CmdDown(); | ||||
| 
 | ||||
|     // mouse anywhere
 | ||||
|     if (evt.Moving()) | ||||
|     if (evt.Moving()) { | ||||
|         m_tooltip = update_hover_state(mouse_pos); | ||||
|     else if (evt.LeftUp()) { | ||||
|         if (m_current == MmuSegmentation) | ||||
|             gizmo_event(SLAGizmoEventType::Moving, mouse_pos, evt.ShiftDown(), evt.AltDown()); | ||||
|     } else if (evt.LeftUp()) { | ||||
|         if (m_mouse_capture.left) { | ||||
|             processed = true; | ||||
|             m_mouse_capture.left = false; | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Lukáš Hejl
						Lukáš Hejl