mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-26 10:11:10 -06:00 
			
		
		
		
	FDM supports gizmo: fixed a crash when trying to paint on the clipping plane
This commit is contained in:
		
							parent
							
								
									9c365da828
								
							
						
					
					
						commit
						feb591782f
					
				
					 1 changed files with 101 additions and 98 deletions
				
			
		|  | @ -308,7 +308,7 @@ bool GLGizmoFdmSupports::gizmo_event(SLAGizmoEventType action, const Vec2d& mous | ||||||
|         const Transform3d& instance_trafo = mi->get_transformation().get_matrix(); |         const Transform3d& instance_trafo = mi->get_transformation().get_matrix(); | ||||||
| 
 | 
 | ||||||
|         std::vector<std::vector<std::pair<Vec3f, size_t>>> hit_positions_and_facet_ids; |         std::vector<std::vector<std::pair<Vec3f, size_t>>> hit_positions_and_facet_ids; | ||||||
|         bool some_mesh_was_hit = false; |         bool clipped_mesh_was_hit = false; | ||||||
| 
 | 
 | ||||||
|         Vec3f normal =  Vec3f::Zero(); |         Vec3f normal =  Vec3f::Zero(); | ||||||
|         Vec3f hit = Vec3f::Zero(); |         Vec3f hit = Vec3f::Zero(); | ||||||
|  | @ -343,7 +343,7 @@ bool GLGizmoFdmSupports::gizmo_event(SLAGizmoEventType action, const Vec2d& mous | ||||||
|             { |             { | ||||||
|                 // In case this hit is clipped, skip it.
 |                 // In case this hit is clipped, skip it.
 | ||||||
|                 if (is_mesh_point_clipped(hit.cast<double>())) { |                 if (is_mesh_point_clipped(hit.cast<double>())) { | ||||||
|                     some_mesh_was_hit = true; |                     clipped_mesh_was_hit = true; | ||||||
|                     continue; |                     continue; | ||||||
|                 } |                 } | ||||||
| 
 | 
 | ||||||
|  | @ -357,116 +357,119 @@ bool GLGizmoFdmSupports::gizmo_event(SLAGizmoEventType action, const Vec2d& mous | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         // We now know where the ray hit, let's save it and cast another ray
 |  | ||||||
|         if (closest_hit_mesh_id != size_t(-1)) // only if there is at least one hit
 |  | ||||||
|             some_mesh_was_hit = true; |  | ||||||
| 
 | 
 | ||||||
|         if (some_mesh_was_hit) { |         if (closest_hit_mesh_id == -1) { | ||||||
|             // Now propagate the hits
 |             // In case we have no valid hit, we can return. The event will
 | ||||||
|             mesh_id = -1; |             // be stopped in following two cases:
 | ||||||
|             const TriangleMesh* mesh = nullptr; |             //  1. clicking the clipping plane
 | ||||||
|             for (const ModelVolume* mv : mo->volumes) { |             //  2. dragging while painting (to prevent scene rotations and moving the object)
 | ||||||
|                 if (! mv->is_model_part()) |             return clipped_mesh_was_hit | ||||||
|                     continue; |                 || (action == SLAGizmoEventType::Dragging && m_button_down != Button::None); | ||||||
|                 ++mesh_id; |         } | ||||||
|                 if (mesh_id == closest_hit_mesh_id) { | 
 | ||||||
|                     mesh = &mv->mesh(); |         // Now propagate the hits
 | ||||||
|                     break; |         mesh_id = -1; | ||||||
|                 } |         const TriangleMesh* mesh = nullptr; | ||||||
|  |         for (const ModelVolume* mv : mo->volumes) { | ||||||
|  |             if (! mv->is_model_part()) | ||||||
|  |                 continue; | ||||||
|  |             ++mesh_id; | ||||||
|  |             if (mesh_id == closest_hit_mesh_id) { | ||||||
|  |                 mesh = &mv->mesh(); | ||||||
|  |                 break; | ||||||
|             } |             } | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|             bool update_both = false; |         bool update_both = false; | ||||||
| 
 | 
 | ||||||
|             const Transform3d& trafo_matrix = trafo_matrices[mesh_id]; |         const Transform3d& trafo_matrix = trafo_matrices[mesh_id]; | ||||||
| 
 | 
 | ||||||
|             // Calculate how far can a point be from the line (in mesh coords).
 |         // Calculate how far can a point be from the line (in mesh coords).
 | ||||||
|             // FIXME: The scaling of the mesh can be non-uniform.
 |         // FIXME: The scaling of the mesh can be non-uniform.
 | ||||||
|             const Vec3d sf = Geometry::Transformation(trafo_matrix).get_scaling_factor(); |         const Vec3d sf = Geometry::Transformation(trafo_matrix).get_scaling_factor(); | ||||||
|             const float avg_scaling = (sf(0) + sf(1) + sf(2))/3.; |         const float avg_scaling = (sf(0) + sf(1) + sf(2))/3.; | ||||||
|             const float limit = pow(m_cursor_radius/avg_scaling , 2.f); |         const float limit = pow(m_cursor_radius/avg_scaling , 2.f); | ||||||
| 
 | 
 | ||||||
|             const std::pair<Vec3f, size_t>& hit_and_facet = { closest_hit, closest_facet }; |         const std::pair<Vec3f, size_t>& hit_and_facet = { closest_hit, closest_facet }; | ||||||
| 
 | 
 | ||||||
|             // Calculate direction from camera to the hit (in mesh coords):
 |         // Calculate direction from camera to the hit (in mesh coords):
 | ||||||
|             Vec3f dir = ((trafo_matrix.inverse() * camera.get_position()).cast<float>() - hit_and_facet.first).normalized(); |         Vec3f dir = ((trafo_matrix.inverse() * camera.get_position()).cast<float>() - hit_and_facet.first).normalized(); | ||||||
| 
 | 
 | ||||||
|             // A lambda to calculate distance from the centerline:
 |         // A lambda to calculate distance from the centerline:
 | ||||||
|             auto squared_distance_from_line = [&hit_and_facet, &dir](const Vec3f& point) -> float { |         auto squared_distance_from_line = [&hit_and_facet, &dir](const Vec3f& point) -> float { | ||||||
|                 Vec3f diff = hit_and_facet.first - point; |             Vec3f diff = hit_and_facet.first - point; | ||||||
|                 return (diff - diff.dot(dir) * dir).squaredNorm(); |             return (diff - diff.dot(dir) * dir).squaredNorm(); | ||||||
|             }; |         }; | ||||||
| 
 | 
 | ||||||
|             // A lambda to determine whether this facet is potentionally visible (still can be obscured)
 |         // A lambda to determine whether this facet is potentionally visible (still can be obscured)
 | ||||||
|             auto faces_camera = [&dir, &mesh](const size_t& facet) -> bool { |         auto faces_camera = [&dir, &mesh](const size_t& facet) -> bool { | ||||||
|                 return (mesh->stl.facet_start[facet].normal.dot(dir) > 0.); |             return (mesh->stl.facet_start[facet].normal.dot(dir) > 0.); | ||||||
|             }; |         }; | ||||||
|             // Now start with the facet the pointer points to and check all adjacent facets.
 |         // Now start with the facet the pointer points to and check all adjacent facets.
 | ||||||
|             std::vector<size_t> facets_to_select{hit_and_facet.second}; |         std::vector<size_t> facets_to_select{hit_and_facet.second}; | ||||||
|             std::vector<bool> visited(m_selected_facets[mesh_id].size(), false); // keep track of facets we already processed
 |         std::vector<bool> visited(m_selected_facets[mesh_id].size(), false); // keep track of facets we already processed
 | ||||||
|             size_t facet_idx = 0; // index into facets_to_select
 |         size_t facet_idx = 0; // index into facets_to_select
 | ||||||
|             while (facet_idx < facets_to_select.size()) { |         while (facet_idx < facets_to_select.size()) { | ||||||
|                 size_t facet = facets_to_select[facet_idx]; |             size_t facet = facets_to_select[facet_idx]; | ||||||
|                 if (! visited[facet]) { |             if (! visited[facet]) { | ||||||
|                     // check all three vertices and in case they're close enough,
 |                 // check all three vertices and in case they're close enough,
 | ||||||
|                     // add neighboring facets to be proccessed later
 |                 // add neighboring facets to be proccessed later
 | ||||||
|                     for (size_t i=0; i<3; ++i) { |                 for (size_t i=0; i<3; ++i) { | ||||||
|                         float dist = squared_distance_from_line( |                     float dist = squared_distance_from_line( | ||||||
|                                     mesh->its.vertices[mesh->its.indices[facet](i)]); |                                 mesh->its.vertices[mesh->its.indices[facet](i)]); | ||||||
|                         if (dist < limit) { |                     if (dist < limit) { | ||||||
|                             for (int n=0; n<3; ++n) { |                         for (int n=0; n<3; ++n) { | ||||||
|                                 if (faces_camera(mesh->stl.neighbors_start[facet].neighbor[n])) |                             if (faces_camera(mesh->stl.neighbors_start[facet].neighbor[n])) | ||||||
|                                     facets_to_select.push_back(mesh->stl.neighbors_start[facet].neighbor[n]); |                                 facets_to_select.push_back(mesh->stl.neighbors_start[facet].neighbor[n]); | ||||||
|                             } |  | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                     visited[facet] = true; |  | ||||||
|                 } |                 } | ||||||
|                 ++facet_idx; |                 visited[facet] = true; | ||||||
|             } |             } | ||||||
| 
 |             ++facet_idx; | ||||||
|             std::vector<size_t> new_facets; |  | ||||||
|             new_facets.reserve(facets_to_select.size()); |  | ||||||
| 
 |  | ||||||
|             // Now just select all facets that passed and remember which
 |  | ||||||
|             // ones have really changed state.
 |  | ||||||
|             for (size_t next_facet : facets_to_select) { |  | ||||||
|                 FacetSupportType& facet = m_selected_facets[mesh_id][next_facet]; |  | ||||||
| 
 |  | ||||||
|                 if (facet != new_state) { |  | ||||||
|                     if (facet != FacetSupportType::NONE) { |  | ||||||
|                         // this triangle is currently in the other VBA.
 |  | ||||||
|                         // Both VBAs need to be refreshed.
 |  | ||||||
|                         update_both = true; |  | ||||||
|                     } |  | ||||||
|                     facet = new_state; |  | ||||||
|                     new_facets.push_back(next_facet); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (! new_facets.empty()) { |  | ||||||
|                 if (new_state != FacetSupportType::NONE) { |  | ||||||
|                     // append triangles into the respective VBA
 |  | ||||||
|                     update_vertex_buffers(mesh, mesh_id, new_state, &new_facets); |  | ||||||
|                     if (update_both) { |  | ||||||
|                         auto other = new_state == FacetSupportType::ENFORCER |  | ||||||
|                                 ? FacetSupportType::BLOCKER |  | ||||||
|                                 : FacetSupportType::ENFORCER; |  | ||||||
|                         update_vertex_buffers(mesh, mesh_id, other); // regenerate the other VBA
 |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 else { |  | ||||||
|                     update_vertex_buffers(mesh, mesh_id, FacetSupportType::ENFORCER); |  | ||||||
|                     update_vertex_buffers(mesh, mesh_id, FacetSupportType::BLOCKER); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
|             if (m_button_down == Button::None) |  | ||||||
|                 m_button_down = ((action == SLAGizmoEventType::LeftDown) ? Button::Left : Button::Right); |  | ||||||
|             return true; |  | ||||||
|         } |         } | ||||||
|         if (action == SLAGizmoEventType::Dragging && m_button_down != Button::None) | 
 | ||||||
|             return true; |         std::vector<size_t> new_facets; | ||||||
|  |         new_facets.reserve(facets_to_select.size()); | ||||||
|  | 
 | ||||||
|  |         // Now just select all facets that passed and remember which
 | ||||||
|  |         // ones have really changed state.
 | ||||||
|  |         for (size_t next_facet : facets_to_select) { | ||||||
|  |             FacetSupportType& facet = m_selected_facets[mesh_id][next_facet]; | ||||||
|  | 
 | ||||||
|  |             if (facet != new_state) { | ||||||
|  |                 if (facet != FacetSupportType::NONE) { | ||||||
|  |                     // this triangle is currently in the other VBA.
 | ||||||
|  |                     // Both VBAs need to be refreshed.
 | ||||||
|  |                     update_both = true; | ||||||
|  |                 } | ||||||
|  |                 facet = new_state; | ||||||
|  |                 new_facets.push_back(next_facet); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (! new_facets.empty()) { | ||||||
|  |             if (new_state != FacetSupportType::NONE) { | ||||||
|  |                 // append triangles into the respective VBA
 | ||||||
|  |                 update_vertex_buffers(mesh, mesh_id, new_state, &new_facets); | ||||||
|  |                 if (update_both) { | ||||||
|  |                     auto other = new_state == FacetSupportType::ENFORCER | ||||||
|  |                             ? FacetSupportType::BLOCKER | ||||||
|  |                             : FacetSupportType::ENFORCER; | ||||||
|  |                     update_vertex_buffers(mesh, mesh_id, other); // regenerate the other VBA
 | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             else { | ||||||
|  |                 update_vertex_buffers(mesh, mesh_id, FacetSupportType::ENFORCER); | ||||||
|  |                 update_vertex_buffers(mesh, mesh_id, FacetSupportType::BLOCKER); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         if (m_button_down == Button::None) | ||||||
|  |             m_button_down = ((action == SLAGizmoEventType::LeftDown) ? Button::Left : Button::Right); | ||||||
|  | 
 | ||||||
|  |         return true; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if ((action == SLAGizmoEventType::LeftUp || action == SLAGizmoEventType::RightUp) |     if ((action == SLAGizmoEventType::LeftUp || action == SLAGizmoEventType::RightUp) | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Lukas Matena
						Lukas Matena