mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-25 17:51:10 -06:00 
			
		
		
		
	TriangleSelector: Reusing of previously calculated triangle divisions, partial garbage collection implementation
This commit is contained in:
		
							parent
							
								
									fb73bb1c66
								
							
						
					
					
						commit
						b9321856f3
					
				
					 2 changed files with 117 additions and 65 deletions
				
			
		|  | @ -800,51 +800,29 @@ void GLGizmoFdmSupports::on_save(cereal::BinaryOutputArchive&) const | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| 
 | // sides_to_split==-1 : just restore previous split
 | ||||||
| void TriangleSelector::Triangle::set_division(int sides_to_split, int special_side_idx) | void TriangleSelector::Triangle::set_division(int sides_to_split, int special_side_idx) | ||||||
| { | { | ||||||
|     assert(sides_to_split >=0 && sides_to_split <= 3); |     assert(sides_to_split >=-1 && sides_to_split <= 3); | ||||||
|     assert(special_side_idx >=-1 && special_side_idx < 3); |     assert(special_side_idx >=-1 && special_side_idx < 3); | ||||||
| 
 | 
 | ||||||
|     // If splitting one or two sides, second argument must be provided.
 |     // If splitting one or two sides, second argument must be provided.
 | ||||||
|     assert(sides_to_split != 1 || special_side_idx != -1); |     assert(sides_to_split != 1 || special_side_idx != -1); | ||||||
|     assert(sides_to_split != 2 || special_side_idx != -1); |     assert(sides_to_split != 2 || special_side_idx != -1); | ||||||
| 
 | 
 | ||||||
|     division_type = sides_to_split | ((special_side_idx != -1 ? special_side_idx : 0 ) <<2); |     if (sides_to_split != -1) { | ||||||
| } |         this->number_of_splits = sides_to_split; | ||||||
| 
 |         if (sides_to_split != 0) { | ||||||
| 
 |             assert(old_number_of_splits == 0); | ||||||
| 
 |             this->special_side_idx = special_side_idx; | ||||||
| void TriangleSelector::Triangle::set_state(FacetSupportType type) |             this->old_number_of_splits = sides_to_split; | ||||||
| { |         } | ||||||
|     // If this is not a leaf-node, this makes no sense and
 |     } | ||||||
|     // the bits are used for storing index of an edge.
 |     else { | ||||||
|     assert(! is_split()); |         assert(old_number_of_splits != 0); | ||||||
|     division_type = (int8_t(type) << 2); |         this->number_of_splits = old_number_of_splits; | ||||||
| } |         // indices of children should still be there.
 | ||||||
| 
 |     } | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| int TriangleSelector::Triangle::side_to_keep() const |  | ||||||
| { |  | ||||||
|     assert(number_of_split_sides() == 2); |  | ||||||
|     return division_type >> 2; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| int TriangleSelector::Triangle::side_to_split() const |  | ||||||
| { |  | ||||||
|     assert(number_of_split_sides() == 1); |  | ||||||
|     return division_type >> 2; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| FacetSupportType TriangleSelector::Triangle::get_state() const |  | ||||||
| { |  | ||||||
|     assert(! is_split()); // this must be leaf
 |  | ||||||
|     return FacetSupportType(division_type >> 2); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -938,17 +916,29 @@ bool TriangleSelector::select_triangle(int facet_idx, FacetSupportType type, boo | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| bool TriangleSelector::split_triangle(int facet_idx) | void TriangleSelector::split_triangle(int facet_idx) | ||||||
| { | { | ||||||
|     if (m_triangles[facet_idx].is_split()) { |     if (m_triangles[facet_idx].is_split()) { | ||||||
|         // The triangle was divided already.
 |         // The triangle is divided already.
 | ||||||
|         return false; |         return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     Triangle* tr = &m_triangles[facet_idx]; |     Triangle* tr = &m_triangles[facet_idx]; | ||||||
| 
 | 
 | ||||||
|     FacetSupportType old_type = tr->get_state(); |     FacetSupportType old_type = tr->get_state(); | ||||||
| 
 | 
 | ||||||
|  |     if (tr->was_split_before() != 0) { | ||||||
|  |         // This triangle is not split at the moment, but was at one point
 | ||||||
|  |         // in history. We can just restore it and resurrect its children.
 | ||||||
|  |         tr->set_division(-1); | ||||||
|  |         for (int i=0; i<=tr->number_of_split_sides(); ++i) { | ||||||
|  |             m_triangles[tr->children[i]].set_state(old_type); | ||||||
|  |             m_triangles[tr->children[i]].valid = true; | ||||||
|  |         } | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // If we got here, we are about to actually split the triangle.
 | ||||||
|     const double limit_squared = m_edge_limit_sqr; |     const double limit_squared = m_edge_limit_sqr; | ||||||
| 
 | 
 | ||||||
|     stl_triangle_vertex_indices& facet = tr->verts_idxs; |     stl_triangle_vertex_indices& facet = tr->verts_idxs; | ||||||
|  | @ -964,8 +954,9 @@ bool TriangleSelector::split_triangle(int facet_idx) | ||||||
|             side_to_keep = pt_idx; |             side_to_keep = pt_idx; | ||||||
|     } |     } | ||||||
|     if (sides_to_split.empty()) { |     if (sides_to_split.empty()) { | ||||||
|  |         // This shall be unselected.
 | ||||||
|         tr->set_division(0); |         tr->set_division(0); | ||||||
|         return false; |         return; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     // indices of triangle vertices
 |     // indices of triangle vertices
 | ||||||
|  | @ -1023,11 +1014,8 @@ bool TriangleSelector::split_triangle(int facet_idx) | ||||||
|     assert(! sides_to_split.empty() && int(sides_to_split.size()) <= 3); |     assert(! sides_to_split.empty() && int(sides_to_split.size()) <= 3); | ||||||
|     for (int i=0; i<=int(sides_to_split.size()); ++i) { |     for (int i=0; i<=int(sides_to_split.size()); ++i) { | ||||||
|         tr->children[i] = m_triangles.size()-1-i; |         tr->children[i] = m_triangles.size()-1-i; | ||||||
|         m_triangles[tr->children[i]].parent = facet_idx; |  | ||||||
|         m_triangles[tr->children[i]].set_state(old_type); |         m_triangles[tr->children[i]].set_state(old_type); | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     return true; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -1166,6 +1154,45 @@ void TriangleSelector::remove_useless_children(int facet_idx) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | void TriangleSelector::garbage_collect() | ||||||
|  | { | ||||||
|  |     // First make a map from old to new triangle indices.
 | ||||||
|  |     int new_idx = m_orig_size_indices; | ||||||
|  |     std::vector<int> new_triangle_indices(m_triangles.size(), -1); | ||||||
|  |     std::vector<bool> invalid_vertices(m_vertices.size(), false); | ||||||
|  |     for (int i = m_orig_size_indices; i<int(m_triangles.size()); ++i) { | ||||||
|  |         if (m_triangles[i].valid) { | ||||||
|  |             new_triangle_indices[i] = new_idx; | ||||||
|  |             ++new_idx; | ||||||
|  |         } else { | ||||||
|  |             // FIXME: Decrement reference counter for the vertices.
 | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     // We can remove all invalid triangles and vertices that are no longer referenced.
 | ||||||
|  |     m_triangles.erase(std::remove_if(m_triangles.begin()+m_orig_size_indices, m_triangles.end(), | ||||||
|  |                           [](const Triangle& tr) { return ! tr.valid; }), | ||||||
|  |                       m_triangles.end()); | ||||||
|  | 
 | ||||||
|  |     // Now go through all remaining triangles and update changed indices.
 | ||||||
|  |     for (Triangle& tr : m_triangles) { | ||||||
|  |         assert(tr.valid); | ||||||
|  | 
 | ||||||
|  |         if (tr.is_split()) { | ||||||
|  |             // There are children. Update their indices.
 | ||||||
|  |             for (int j=0; j<=tr.number_of_split_sides(); ++j) { | ||||||
|  |                 assert(new_triangle_indices[tr.children[j]] != -1); | ||||||
|  |                 tr.children[j] = new_triangle_indices[tr.children[j]]; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         // If this triangle was split before, forget it.
 | ||||||
|  |         // Children referenced in the cache are dead by now.
 | ||||||
|  |         tr.forget_history(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| TriangleSelector::TriangleSelector(const TriangleMesh& mesh) | TriangleSelector::TriangleSelector(const TriangleMesh& mesh) | ||||||
| { | { | ||||||
|     for (const stl_vertex& vert : mesh.its.vertices) |     for (const stl_vertex& vert : mesh.its.vertices) | ||||||
|  | @ -1209,6 +1236,20 @@ void TriangleSelector::render(ImGuiWrapper* imgui) | ||||||
| #endif | #endif | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | void TriangleSelector::set_edge_limit(float edge_limit) | ||||||
|  | { | ||||||
|  |     float new_limit_sqr = std::pow(edge_limit, 2.f); | ||||||
|  | 
 | ||||||
|  |     if (new_limit_sqr != m_edge_limit_sqr) { | ||||||
|  |         m_edge_limit_sqr = new_limit_sqr; | ||||||
|  | 
 | ||||||
|  |         // The way how triangles split may be different now, forget
 | ||||||
|  |         // all cached splits.
 | ||||||
|  |         garbage_collect(); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #ifdef PRUSASLICER_TRIANGLE_SELECTOR_DEBUG | #ifdef PRUSASLICER_TRIANGLE_SELECTOR_DEBUG | ||||||
| void TriangleSelector::render_debug(ImGuiWrapper* imgui) | void TriangleSelector::render_debug(ImGuiWrapper* imgui) | ||||||
| { | { | ||||||
|  | @ -1218,28 +1259,35 @@ void TriangleSelector::render_debug(ImGuiWrapper* imgui) | ||||||
|     imgui->text("Edge limit (mm): "); |     imgui->text("Edge limit (mm): "); | ||||||
|     imgui->slider_float("", &edge_limit, 0.1f, 8.f); |     imgui->slider_float("", &edge_limit, 0.1f, 8.f); | ||||||
|     set_edge_limit(edge_limit); |     set_edge_limit(edge_limit); | ||||||
|     imgui->checkbox("Show triangles: ", m_show_triangles); |     imgui->checkbox("Show split triangles: ", m_show_triangles); | ||||||
|  |     imgui->checkbox("Show invalid triangles: ", m_show_invalid); | ||||||
| 
 | 
 | ||||||
|     int valid_triangles = std::count_if(m_triangles.begin(), m_triangles.end(), |     int valid_triangles = std::count_if(m_triangles.begin(), m_triangles.end(), | ||||||
|                                 [](const Triangle& tr) { return tr.valid; }); |                                 [](const Triangle& tr) { return tr.valid; }); | ||||||
|     imgui->text("Valid triangles: " + std::to_string(valid_triangles) + |     imgui->text("Valid triangles: " + std::to_string(valid_triangles) + | ||||||
|                   "/" + std::to_string(m_triangles.size())); |                   "/" + std::to_string(m_triangles.size())); | ||||||
|     imgui->text("Number of vertices: " + std::to_string(m_vertices.size())); |     imgui->text("Number of vertices: " + std::to_string(m_vertices.size())); | ||||||
|  |     if (imgui->button("Force garbage collection")) | ||||||
|  |         garbage_collect(); | ||||||
| 
 | 
 | ||||||
|     imgui->end(); |     imgui->end(); | ||||||
| 
 | 
 | ||||||
|     if (m_show_triangles) { |     if (m_show_triangles) { | ||||||
|         ::glColor3f(0.f, 0.f, 1.f); |  | ||||||
|         ::glPolygonMode( GL_FRONT_AND_BACK, GL_LINE ); |         ::glPolygonMode( GL_FRONT_AND_BACK, GL_LINE ); | ||||||
| 
 | 
 | ||||||
|         ::glBegin( GL_TRIANGLES); |         ::glBegin( GL_TRIANGLES); | ||||||
|         for (int tr_id=0; tr_id<int(m_triangles.size()); ++tr_id) { |         for (int tr_id=0; tr_id<int(m_triangles.size()); ++tr_id) { | ||||||
|             const Triangle& tr = m_triangles[tr_id]; |             const Triangle& tr = m_triangles[tr_id]; | ||||||
|             if (! tr.valid) |             if (! m_show_invalid && ! tr.valid) | ||||||
|                 continue; |                 continue; | ||||||
| 
 | 
 | ||||||
|             if (tr_id == m_orig_size_indices-1) |             if (tr.valid) | ||||||
|                 ::glColor3f(1.f, 0.f, 0.f); |                 ::glColor3f(1.f, 0.f, 0.f); | ||||||
|  |             else | ||||||
|  |                 ::glColor3f(1.f, 1.f, 0.f); | ||||||
|  | 
 | ||||||
|  |             if (tr_id < m_orig_size_indices) | ||||||
|  |                 ::glColor3f(0.f, 0.f, 1.f); | ||||||
| 
 | 
 | ||||||
|             for (int i=0; i<3; ++i) |             for (int i=0; i<3; ++i) | ||||||
|                 ::glVertex3f(m_vertices[tr.verts_idxs[i]][0], |                 ::glVertex3f(m_vertices[tr.verts_idxs[i]][0], | ||||||
|  |  | ||||||
|  | @ -27,7 +27,7 @@ class ClippingPlane; | ||||||
| // to recursively subdivide the triangles and make the selection finer.
 | // to recursively subdivide the triangles and make the selection finer.
 | ||||||
| class TriangleSelector { | class TriangleSelector { | ||||||
| public: | public: | ||||||
|     void set_edge_limit(float edge_limit) { m_edge_limit_sqr = std::pow(edge_limit, 2.f); } |     void set_edge_limit(float edge_limit); | ||||||
| 
 | 
 | ||||||
|     // Create new object on a TriangleMesh. The referenced mesh must
 |     // Create new object on a TriangleMesh. The referenced mesh must
 | ||||||
|     // stay valid, a ptr to it is saved and used.
 |     // stay valid, a ptr to it is saved and used.
 | ||||||
|  | @ -54,6 +54,7 @@ public: | ||||||
| #ifdef PRUSASLICER_TRIANGLE_SELECTOR_DEBUG | #ifdef PRUSASLICER_TRIANGLE_SELECTOR_DEBUG | ||||||
|     void render_debug(ImGuiWrapper* imgui); |     void render_debug(ImGuiWrapper* imgui); | ||||||
|     bool m_show_triangles{true}; |     bool m_show_triangles{true}; | ||||||
|  |     bool m_show_invalid{false}; | ||||||
| #endif | #endif | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  | @ -62,16 +63,16 @@ private: | ||||||
|     public: |     public: | ||||||
|         Triangle(int a, int b, int c) |         Triangle(int a, int b, int c) | ||||||
|             : verts_idxs{stl_triangle_vertex_indices(a, b, c)}, |             : verts_idxs{stl_triangle_vertex_indices(a, b, c)}, | ||||||
|               division_type{0} |               state{FacetSupportType(0)}, | ||||||
|  |               number_of_splits{0}, | ||||||
|  |               special_side_idx{0}, | ||||||
|  |               old_number_of_splits{0} | ||||||
|         {} |         {} | ||||||
|         stl_triangle_vertex_indices verts_idxs; |         stl_triangle_vertex_indices verts_idxs; | ||||||
| 
 | 
 | ||||||
|         // Is this triangle valid or marked to remove?
 |         // Is this triangle valid or marked to remove?
 | ||||||
|         bool valid{true}; |         bool valid{true}; | ||||||
| 
 | 
 | ||||||
|         // Index of parent triangle (-1: original)
 |  | ||||||
|         int parent{-1}; |  | ||||||
| 
 |  | ||||||
|         // Children triangles (0 = no child)
 |         // Children triangles (0 = no child)
 | ||||||
|         std::array<int, 4> children; |         std::array<int, 4> children; | ||||||
| 
 | 
 | ||||||
|  | @ -79,22 +80,25 @@ private: | ||||||
|         void set_division(int sides_to_split, int special_side_idx = -1); |         void set_division(int sides_to_split, int special_side_idx = -1); | ||||||
| 
 | 
 | ||||||
|         // Get/set current state.
 |         // Get/set current state.
 | ||||||
|         void set_state(FacetSupportType state); |         void set_state(FacetSupportType type) { assert(! is_split()); state = type; } | ||||||
|         FacetSupportType get_state() const; |         FacetSupportType get_state() const { assert(! is_split()); return state; } | ||||||
| 
 | 
 | ||||||
|         // Get info on how it's split.
 |         // Get info on how it's split.
 | ||||||
|         bool is_split() const { return number_of_split_sides() != 0; } |         bool is_split() const { return number_of_split_sides() != 0; } | ||||||
|         int number_of_split_sides() const { return division_type & 0b11; } |         int number_of_split_sides() const { return number_of_splits; } | ||||||
|         int side_to_keep() const; |         int side_to_keep() const  { assert(number_of_split_sides() == 2); return special_side_idx; } | ||||||
|         int side_to_split() const; |         int side_to_split() const { assert(number_of_split_sides() == 1); return special_side_idx; } | ||||||
|  |         bool was_split_before() const { return old_number_of_splits != 0; } | ||||||
|  |         void forget_history() { old_number_of_splits = 0; } | ||||||
| 
 | 
 | ||||||
|     private: |     private: | ||||||
|         // Bitmask encoding which sides are split.
 |         int number_of_splits; | ||||||
|         int8_t division_type; |         int special_side_idx; | ||||||
|         // bits 0, 1 : decimal 0, 1, 2 or 3 (how many sides are split)
 |         FacetSupportType state; | ||||||
|         // bits 2, 3 (non-leaf): decimal 0, 1 or 2 identifying the special edge
 | 
 | ||||||
|         //   (one that splits in one-edge split or one that stays in two-edge split).
 |         // How many children were spawned during last split?
 | ||||||
|         // bits 2, 3 (leaf): FacetSupportType value
 |         // Is not reset on remerging the triangle.
 | ||||||
|  |         int old_number_of_splits; | ||||||
|     }; |     }; | ||||||
| 
 | 
 | ||||||
|     // Lists of vertices and triangles, both original and new
 |     // Lists of vertices and triangles, both original and new
 | ||||||
|  | @ -129,7 +133,7 @@ private: | ||||||
|     int vertices_inside(int facet_idx) const; |     int vertices_inside(int facet_idx) const; | ||||||
|     bool faces_camera(int facet) const; |     bool faces_camera(int facet) const; | ||||||
|     void undivide_triangle(int facet_idx); |     void undivide_triangle(int facet_idx); | ||||||
|     bool split_triangle(int facet_idx); |     void split_triangle(int facet_idx); | ||||||
|     void remove_useless_children(int facet_idx); // No hidden meaning. Triangles are meant.
 |     void remove_useless_children(int facet_idx); // No hidden meaning. Triangles are meant.
 | ||||||
|     bool is_pointer_in_triangle(int facet_idx) const; |     bool is_pointer_in_triangle(int facet_idx) const; | ||||||
|     bool is_edge_inside_cursor(int facet_idx) const; |     bool is_edge_inside_cursor(int facet_idx) const; | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Lukas Matena
						Lukas Matena